我为getaddrinfo写了一个非常简单的测试程序:
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <stdio.h>
#include <string.h>
int main() {
struct addrinfo hints;
struct addrinfo *res, *rp;
char hoststr[64], servstr[8];
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
getaddrinfo(NULL, "9998", &hints, &res);
for (rp = res; rp != NULL; rp = rp->ai_next) {
getnameinfo(rp->ai_addr, rp->ai_addrlen, hoststr, sizeof(hoststr),
servstr, sizeof(servstr), NI_NUMERICHOST | NI_NUMERICSERV);
printf("%s:%s\n", hoststr, servstr);
}
}
当我编译并运行此程序时,它会在IPv6地址之前提供IPv4地址:
# gcc -o getaddrinfo getaddrinfo.c
# ./getaddrinfo
0.0.0.0:9998
:::9998
据我所知,其他来源的IPv6地址应优先于IPv4。我使用默认的/etc/gai.conf,这意味着IPv6应该优先于IPv4。那么为什么getaddrinfo会这样排序呢?
# ip addr show lo
1: lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default
link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00
inet 127.0.0.1/8 scope host lo
valid_lft forever preferred_lft forever
inet6 ::1/128 scope host
valid_lft forever preferred_lft forever
答案 0 :(得分:1)
从手册页:
如果在hints.ai_flags中指定了AI_PASSIVE标志,并且节点是 NULL,则返回的套接字地址将适合 绑定(2)将接受(2)连接的套接字。归来了 套接字地址将包含&#34;通配符地址&#34; (IPv4的INADDR_ANY 地址,IPv6地址IN6ADDR_ANY_INIT)。通配符地址是 由打算接受任何主机网络地址上的连接的应用程序(通常是服务器)使用。如果node不是NULL, 然后忽略AI_PASSIVE标志。
您描述的行为是指您使用getaddrinfo()
查找名称以准备传出连接时的行为。因为您正在反向查找本地地址以绑定服务,所以提供地址的顺序无关紧要。