主机地址是网络通信最重要的数据之一,net 包中定义了三种类型的主机地址数据类型:IP、IPMask 和 IPAddr,它们分别用来存储协议相关的网络地址。
在 net 包中,IP 地址类型被定义成一个 byte 型数组,即若干个 8 位组,格式如下:
在 net 包中,有几个函数可以将 IP 地址类型作为函数的返回类型,比如 ParseIP() 函数,该函数原型定义如下:
ParseIP() 函数的主要作用是分析 IP 地址的合法性,如果是一个合法的 IP 地址,ParseIP() 函数将返回一个 IP 地址对象。如果是一个非法 IP 地址,ParseIP() 函数将返回 nil。
还可以使用 IP 对象的 String() 方法将 IP 地址转换成字符串格式,String() 方法的原型定义如下:
如果是 IPv4 地址,String() 方法将返回一个点分十进制格式的 IP 地址,如“192.168.0.1”。如果是 IPv6 地址,String() 方法将返回使用“:”分隔的地址形式,如“2000:0:0:0:0:0:0:1”。另外注意一个特例,对于地址“0:0:0:0:0:0:0:1”的返回结果是省略格式“::1”。
【示例 1】IP 地址类型。
import(
"fmt"
"net"
"os"
)
func main() {
if len(os.Args) != 2 {
fmt.Fprintf(os.Stderr, "Usage: %s ip.addr\n", os.Args[0])
os.Exit(1)
}
addr := os.Args[1]
ip := net.ParseIP(addr)
if ip == nil {
fmt.Println("Invalid address")
} else {
fmt.Println("The address is", ip.String())
}
os.Exit(0)
}
编译并运行该程序,测试过程如下:
在 Go语言中,为了方便子网掩码操作与计算,net 包中还提供了 IPMask 地址类型。在前面讲过,子网掩码地址其实就是一个特殊的 IP 地址,所以 IPMask 类型也是一个 byte 型数组,格式如下:
函数 IPv4Mask() 可以通过一个 32 位 IPv4 地址生成子网掩码地址,调用成功后返回一个 4 字节的十六进制子网掩码地址。IPv4Mask() 函数原型定义如下:
另外,还可以使用主机地址对象的 DefaultMask() 方法获取主机默认子网掩码地址,DefaultMask() 方法原型定义如下:
要注意的是,只有 IPv4 地址才有默认子网掩码。如果不是 IPv4 地址,DefaultMask() 方法将返回 nil。不管是通过调用 IPv4Mask() 函数,还是执行 DefaultMask() 方法,获取的子网掩码地址都是十六进制格式的。例如,子网掩码地址“255.255.255.0”的十六进制格式是“ffffffOO”。
主机地址对象还有一个 Mask() 方法,执行 Mask() 方法后,会返回 IP 地址与子网掩码地址相“与”的结果,这个结果即是主机所处的网络的“网络地址”。Mask() 方法原型定义如下:
还可以通过子网掩码对象的 Size() 方法获取掩码位数 (ones) 和掩码总长度 (bits),如果是一个非标准的子网掩码地址,则 Size() 方法将返回“0,0”。Size() 方法的原型定义如下:
【示例 2】子网掩码地址。
// 子网掩码地址
package main
import(
"fmt"
"net"
"os"
)
func main() {
if len(os.Args) != 2 {
fmt.Fprintf(os.Stderr, "Usage: %s ip.addr\n", os.Args[0])
os.Exit(1)
}
dotaddr := os.Args[1]
addr := net.ParseIP(dotaddr)
if addr == nil {
fmt.Println("Invalid address")
}
mask := addr.DefaultMask()
fmt.Println("Subnet mask is: ", mask.String())
network := addr.Mask(mask)
fmt.Println("Network address is: ", network.String())
ones, bits := mask.Size()
fmt.Println("Mask bits: ", ones, "Total bits: ", bits)
os.Exit(0)
}
编译并运行该程序,结果如下所示:
在 net 包中,许多函数或方法调用后返回的是一个指向 IPAddr 结构体的指针,结构体 IPAddr 内只定义了一个 IP 类型的字段,格式如下:
IPAddr 结构体的主要作用是用于域名解析服务 (DNS),例如,函数 ResolveIPAddr() 可以通过主机名解析主机网络地址。ResolveIPAddr() 函数原型定义如下:
在调用 ResolveIPAddr() 函数时,参数 net 表示网络类型,可以是“ip”、“ip4”或“ip6”,参数 addr 可以是 IP 地址或域名,如果是 IPv6 地址则必须使用“[]”括起来。ResolveIPAddr() 函数调用成功后返回指向 IPAddr 结构体的指针,调用失败返回错误类型 error。
【示例 3】DNS 域名解析。
// DNS 域名解析
package main
import(
"fmt"
"net"
"os"
)
func main() {
if len(os.Args) != 2{
fmt.Fprintf(os.Stderr, "Usage: %s hostname\n", os.Args[0])
fmt.Println("Usage: ", os.Args[0], "hostname")
os.Exit(1)
}
name := os.Args[1]
addr, err := net.ResolveIPAddr("ip", name)
if err != nil {
fmt.Println("Resolvtion error", err.Error())
os.Exit(1)
}
fmt.Println("Resolved address is", addr.String())
os.Exit(0)
}
编译并运行该程序,结果如下所示: