我为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地址之前给出了IPv6地址:
# 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
发布于 2018-03-28 16:24:52
从手册页:
如果AI_PASSIVE标志在hints.ai_flags中指定,且节点为NULL,则返回的套接字地址将适合于绑定(2)将接受(2)连接的套接字。返回的套接字地址将包含“通配符地址”(INADDR_ANY表示IPv4地址,IN6ADDR_ANY_INIT表示IPv6地址)。通配符地址由打算在任何主机网络地址上接受连接的应用程序(通常是服务器)使用。如果节点不是NULL,则忽略AI_PASSIVE标志。
您所描述的行为是在您使用getaddrinfo()
查找名称以准备建立一个传出连接时使用的。因为您正在做相反的操作,所以查找一个本地地址来绑定一个服务,地址显示的顺序是无关的。
https://stackoverflow.com/questions/49537694
复制相似问题