为什么 inet_ntop() 和 inet_ntoa() 给出不同的结果?

1 c networking

我正在创建一个 UDP 服务器-客户端程序。客户端请求文件,如果找到,服务器将发送给客户端。基于 Beej 的网络指南,

  • inet_ntoa() 返回静态缓冲区中的点和数字字符串,每次调用函数都会覆盖该缓冲区。
  • inet_ntop() 成功时返回 dst 参数,失败时返回 NULL(并设置 errno)。

该指南提到 ntoa 已被弃用,因此推荐使用 ntop,因为它支持 IPv4 和 IPv6。

在我的代码中,当我使用 function 或 other 时,我得到了不同的结果,我的理解是它们应该抛出相同的结果。我缺少什么吗?任何帮助将不胜感激。

代码:

    //UDP Client
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <errno.h>
    #include <string.h>
    #include <sys/types.h>
    #include <sys/socket.h>
    #include <netinet/in.h>
    #include <arpa/inet.h>
    #include <netdb.h>

    #define MAXBUFLEN 1024
    #define SER_IP "176.180.226.0" 
    #define SER_PORT "1212"

    // Get port, IPv4 or IPv6:
    in_port_t get_in_port(struct sockaddr *sa){
        if (sa->sa_family == AF_INET) {
            return (((struct sockaddr_in*)sa)->sin_port);
        }
        return (((struct sockaddr_in6*)sa)->sin6_port);
    }

    int main(int argc, char *argv[]){
        int sock, rv, numbytes;
        struct addrinfo hints, *servinfo, *p;
        char buffer[MAXBUFLEN];

        memset(&hints, 0, sizeof hints);
        hints.ai_family = AF_UNSPEC;
        hints.ai_socktype = SOCK_DGRAM;

        rv = getaddrinfo(NULL, SER_PORT, &hints, &servinfo);
        if (rv != 0){
            fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
            exit(1);
        }

        // Printing IP, should provide same result
        for(p = servinfo; p != NULL; p = p->ai_next) {
            char str1[INET_ADDRSTRLEN];
            inet_ntop(AF_INET, &p->ai_addr, str1, INET_ADDRSTRLEN); 
            printf("ntop:%s\n", str1) ;
            printf("inet_ntoa:%s \n", inet_ntoa(((struct sockaddr_in *)p->ai_addr)->sin_addr));
            printf("\n");
        }

        exit(1);
    }
Run Code Online (Sandbox Code Playgroud)

电流输出:

ntop:64.80.142.0
inet_ntoa:0.0.0.0 

ntop:160.80.142.0
inet_ntoa:127.0.0.1 
Run Code Online (Sandbox Code Playgroud)

小智 5

根据man页面,在AF_INET参数的情况下src必须指向一个struct in_addr(网络字节顺序)。

struct addrinfo你有一个指针struct sockaddr,基本上是

sa_family_t sa_family;
char        sa_data[];
Run Code Online (Sandbox Code Playgroud)

然而,struct sockaddr_in

sa_family_t    sin_family;
in_port_t      sin_port;
struct in_addr sin_addr;
Run Code Online (Sandbox Code Playgroud)

所以,你需要更换

inet_ntop(AF_INET, &p->ai_addr, str1, INET_ADDRSTRLEN);
Run Code Online (Sandbox Code Playgroud)

通过任一

inet_ntop(AF_INET, &p->ai_addr->sa_data[2], str1, INET_ADDRSTRLEN);
Run Code Online (Sandbox Code Playgroud)

src参数可能是&p->ai_addr->sa_data[1 << 1]为了避免“幻数” 2- 对端口号存储计数的偏移量)

或者

inet_ntop(AF_INET, &((struct sockaddr_in *)p->ai_addr)->sin_addr, str1, INET_ADDRSTRLEN);
Run Code Online (Sandbox Code Playgroud)

然后它将产生正确的输出。