Go中获取域名IP最直接方式是用net.LookupIP,返回[]net.IP需自行过滤IPv4/IPv6;可自定义net.Resolver指定DNS服务器;建议结合context.WithTimeout控制超时。

在 Go 语言中获取域名对应的 IP 地址,最直接的方式是使用标准库 net 包提供的 DNS 解析功能,无需引入第三方依赖。Go 默认使用系统配置的 DNS(如 /etc/resolv.conf)或环境变量指定的 DNS 服务器,也支持自定义 DNS 查询。
使用 net.LookupIP 获取 IPv4/IPv6 地址
net.LookupIP 是最常用的方法,它返回域名对应的所有 IP 地址(包括 IPv4 和 IPv6):
- 返回值为
[]net.IP,需自行过滤 IPv4(ip.To4() != nil)或 IPv6 - 若解析失败(如域名不存在、网络不通),会返回非 nil 的 error
- 底层调用的是系统 getaddrinfo 或内置 DNS 客户端,行为可靠且跨平台
示例:
ips, err := net.LookupIP("google.com")
if err != nil {
log.Fatal(err)
}
for _, ip := range ips {
if ipv4 := ip.To4(); ipv4 != nil {
fmt.Println("IPv4:", ipv4.String())
}
}
分别查询 A 记录(IPv4)和 AAAA 记录(IPv6)
若需明确区分协议类型,可使用更具体的函数:
立即学习“go语言免费学习笔记(深入)”;
-
net.LookupHost("example.com"):返回所有 IP 字符串(不区分 v4/v6) -
net.LookupAddr("8.8.8.8"):反向查询(IP → 域名) -
net.LookupCNAME("www.google.com"):获取规范域名(CNAME)
注意:LookupHost 和 LookupIP 行为相似,但前者返回 []string,后者返回 []net.IP,推荐优先用 LookupIP 便于后续网络操作。
指定 DNS 服务器进行解析(自定义 resolver)
当需要绕过系统 DNS、测试特定服务器或实现 DoH/DoT 时,可使用 net.Resolver 配合自定义 net.DialContext:
- 创建
&net.Resolver{...},设置PreferGo: true强制使用 Go 内置解析器 - 通过
Dial字段指定 UDP/TCP 连接(例如指向1.1.1.1:53) - 简单定制示例(UDP 查询 1.1.1.1):
resolver := &net.Resolver{
PreferGo: true,
Dial: func(ctx context.Context, network, addr string) (net.Conn, error) {
d := net.Dialer{}
return d.DialContext(ctx, "udp", "1.1.1.1:53")
},
}
ips, err := resolver.LookupIP(context.Background(), "ip4", "github.com")
处理超时与并发解析
DNS 解析默认无超时,生产环境建议显式控制:
- 用
context.WithTimeout包裹解析调用,避免阻塞 - 并发解析多个域名时,可用
errgroup或sync.WaitGroup管理 - 注意:Go 的 DNS 解析在多数情况下已自动缓存(受
GODEBUG和 Go 版本影响),但不保证长期有效,业务层仍需做好重试或降级
示例(带超时):
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) defer cancel() ips, err := net.DefaultResolver.LookupIP(ctx, "ip4", "aws.amazon.com")
不复杂但容易忽略细节,掌握 LookupIP、自定义 Resolver 和上下文控制,就能覆盖绝大多数 DNS 解析需求。










