gethostbyname() 可以返回 IPv6 地址吗?

标签 c sockets network-programming posix

我正在阅读 UNIX NETWORK PROGRAMMING,想出了一个关于练习 11.4 的问题,以支持 gethostbyname 的代码用于 IPv6。书中的答案如下:

#include    "unp.h"

int
main(int argc, char **argv)
{
    int                 sockfd, n;
    char                recvline[MAXLINE + 1];
    struct sockaddr_in  servaddr;
    struct sockaddr_in6 servaddr6;
    struct sockaddr     *sa;
    socklen_t           salen;
    struct in_addr      **pptr;
    struct hostent      *hp;
    struct servent      *sp;

    if (argc != 3)
        err_quit("usage: daytimetcpcli3 <hostname> <service>");

    if ( (hp = gethostbyname(argv[1])) == NULL)
        err_quit("hostname error for %s: %s", argv[1], hstrerror(h_errno));

    if ( (sp = getservbyname(argv[2], "tcp")) == NULL)
        err_quit("getservbyname error for %s", argv[2]);

    pptr = (struct in_addr **) hp->h_addr_list;
    for ( ; *pptr != NULL; pptr++) {
        sockfd = Socket(hp->h_addrtype, SOCK_STREAM, 0);

        if (hp->h_addrtype == AF_INET) {
            sa = (SA *) &servaddr;
            salen = sizeof(servaddr);
        } else if (hp->h_addrtype == AF_INET6) {
            sa = (SA *) &servaddr6;
            salen = sizeof(servaddr6);
        } else
            err_quit("unknown addrtype %d", hp->h_addrtype);

        bzero(sa, salen);
        sa->sa_family = hp->h_addrtype;
        sock_set_port(sa, salen, sp->s_port);
        sock_set_addr(sa, salen, *pptr);

        printf("trying %s\n", Sock_ntop(sa, salen));

        if (connect(sockfd, sa, salen) == 0)
            break;      /* success */
        err_ret("connect error");
        close(sockfd);
    }
    if (*pptr == NULL)
        err_quit("unable to connect");

    while ( (n = Read(sockfd, recvline, MAXLINE)) > 0) {
        recvline[n] = 0;    /* null terminate */
        Fputs(recvline, stdout);
    }
    exit(0);
}

根据书籍和其他资源的信息,我发现gethostbyname 不能返回IPv6。所以我的问题是 gethostbyname 是否可以返回有关 IPV6 的信息,或者只是代码错误?

我觉得这题不是要替换gethostbyname,我想知道这段代码的正确性。

最佳答案

gethostbyname() 不再是(当前)POSIX(IEEE Std 1003.1-2008,2016 版)的一部分。

It explicitly mentions :

The obsolescent h_errno external integer, and the obsolescent gethostbyaddr() and gethostbyname() functions are removed, along with the HOST_NOT_FOUND, NO_DATA, NO_RECOVERY, and TRY_AGAIN macros.


最后一个支持 gethostbyname() 的 POSIX 版本是 IEEE Std 1003.1, 2004 Edition, which states :

struct hostent *gethostbyname(const char *name);

[...]

The gethostbyname() function shall return an entry containing addresses of address family AF_INET for the host with name name.

它根本没有提到 AF_INET6


所以从 POSIX 的角度来看你的问题是关于

the correctness of this code.

可以这样回答:

  • 是的,它曾经是正确的,依赖于实现定义/特定的扩展(即支持 AF_INET6).
  • 不,现在它不再正确,因为 gethostbyname() 已从 POSIX 中删除。

关于gethostbyname() 可以返回 IPv6 地址吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45992814/

相关文章:

ios - 通过本地网络从 iOS 设备浏览桌面文件

c - 从 C 文件输入,数据类型

c - 仅在树莓派上使用 libxml2 访问 xmlNode 时出现段错误

python - 使用 basemap 在世界地图上绘制网络/图形(python)

java - 获取网络上可用计算机的 IP 地址? -java

Python 通过套接字传输文件

c - 将单个字符视为 char *

c - 为什么我的链表冒泡排序函数有时会输出错误的结果,有时又看起来像是无限循环?

无法关闭套接字选项 IPV6_V6ONLY

java - 连接来自 BufferedReader 的字符串并传递给 UI 线程