0

我在用着:

struct in_addr ip;                                         
#define HOST_PORT atoi(argv[2])
#define HOST_IP inet_aton(argv[1], &ip)

sockaddr_in sockaddr;
sockaddr.sin_family = AF_INET;
sockaddr.sin_addr.s_addr = htonl(HOST_IP);
sockaddr.sin_port = htons(HOST_PORT);

在cli中获取ip和端口号。然后我绑定、聆听、接受和阅读。

我进行编译,当我使用./server localhost 8080时,一切都按预期工作。

但是如果我使用./server 127.0.0.1 8080./server 192.168.30.1 8080似乎是绑定和监听,没有错误,但我没有收到请求。

4

1 回答 1

1

你在滥用inet_aton().

您正在将返回值分配inet_aton()给您的sockaddr变量,而不是分配输出的in_addr结构inet_aton()返回值不是 IP 地址

inet_aton() 成功返回非零值,失败返回 0。换句话说,它的返回值基本上是一个布尔值。

使用(aka )bind()的 IP 地址调用指示它将套接字绑定到指定端口上的所有本地网络接口,这对于服务器来说是非常常见的(并且通常是首选)。0.0.0.0INADDR_ANYAF_INET

调用inet_aton("localhost") 失败,因为您无法将主机名传递给inet_aton(),因此它返回0,然后您使用设置为的bind()套接字,这没关系,因此您的代码似乎可以使用,但只是侥幸。sockaddr_ins_addrINADDR_ANY"localhost"

调用inet_aton("127.0.0.1")orinet_aton("192.168.30.1") 成功,因此它返回非零,然后您使用设置为无效 IP 的套接字,导致完全bind()失败,或者更糟的是绑定到与您期望的不同的现有网络接口.sockaddr_ins_addrbind()

此外,in_addr那个inet_aton()输出已经是网络字节顺序,所以不需要调用htonl()它的输出。

试试这个:

struct in_addr ip;
#define HOST_PORT atoi(argv[2])
#define HOST_IP ((inet_aton(argv[1], &ip) != 0) ? ip.s_addr : 0)

struct sockaddr_in sockaddr;
memset(&sockaddr, 0, sizeof(sockaddr));
sockaddr.sin_family = AF_INET;
sockaddr.sin_addr.s_addr = HOST_IP;
sockaddr.sin_port = htons(HOST_PORT);

// use sockaddr as needed ...

或者,干脆去掉你的HOST_IPHOST_PORT宏,因为它们并没有真正帮助你:

struct sockaddr_in sockaddr;

memset(&sockaddr, 0, sizeof(sockaddr));
sockaddr.sin_family = AF_INET;
inet_aton(argv[1], &(sockaddr.sin_addr));
sockaddr.sin_port = htons(atoi(argv[2]));

// use sockaddr as needed ...

话虽这么说,你真的应该使用getaddrinfo(),例如:

struct addrinfo hints, *res;
memset(&hints, 0, sizeof(hints));
hints.ai_flags = AI_PASSIVE;
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_protocol = IPPROTO_TCP;

if (getaddrinfo(argv[1], argv[2], &hints, &res) == 0)
{
    // use res->ai_addr and res->ai_addrlen with socket() and bind() as needed ...
    freeaddrinfo(res);
}

getaddrinfo()会将 IP 地址字符串和主机名字符串解析为适合sockaddr_...您的结构,并愉快地处理与本地主机相关的字符串,如"localhost", "127.0.0.1","192.168.30.1"等。

于 2020-07-06T21:54:19.500 回答