如何在Golang中实现DNS解析_获取域名对应的IP地址

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)

若需明确区分协议类型,可使用更具体的函数:

  • net.LookupHost("example.com"):返回所有 IP 字符串(不区分 v4/v6)
  • net.LookupAddr("8.8.8.8"):反向查询(IP → 域名)
  • net.LookupCNAME("www.google.com"):获取规范域名(CNAME)

注意:LookupHostLookupIP 行为相似,但前者返回 []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 包裹解析调用,避免阻塞
  • 并发解析多个域名时,可用 errgroupsync.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 解析需求。