繁体   English   中英

理解 INADDR_ANY 进行套接字编程

[英]Understanding INADDR_ANY for socket programming

我正在尝试编写一些套接字,因此在服务器端,我使用htonl(INADDR_ANY) 在我理解的范围内,在我看来,这个函数会生成一个随机 IP(我说得对吗?)。 事实上,我想将我的套接字与我的localhost绑定。 但是如果我运行这个

printf("%d",htonl(INADDR_ANY));

我得到 0 作为返回值。 有人可以带来一些解释吗?

  1. INADDR_ANY bind()不会“生成随机 IP”。 它将套接字绑定到所有可用的接口

  2. 对于服务器,您通常希望绑定到所有接口——而不仅仅是“本地主机”。

  3. 如果您只想将套接字绑定到本地主机,则语法为my_sockaddress.sin_addr.s_addr = inet_addr("127.0.0.1"); ,然后调用bind(my_socket, (SOCKADDR *) &my_sockaddr, ...)

  4. 碰巧, INADDR_ANY是一个常数,恰好等于“零”:

    http://www.castaglia.org/proftpd/doc/devel-guide/src/include/inet.h.html

     # define INADDR_ANY ((unsigned long int) 0x00000000) ... # define INADDR_NONE 0xffffffff ... # define INPORT_ANY 0 ...
  5. 如果您还不熟悉它,我强烈建议您查看 Beej 的套接字编程指南:

    http://beej.us/guide/bgnet/

由于人们仍在阅读本文,因此请补充说明:

男人 (7) ip :

当一个进程想要接收新的传入数据包或连接时,它应该使用bind(2)将套接字绑定到本地接口地址。

在这种情况下,只有一个 IP 套接字可以绑定到任何给定的本地(地址、端口)对。 当在绑定调用中指定 INADDR_ANY 时,套接字将绑定到所有本地接口。

当在未绑定的套接字上调用listen(2) 时,该套接字将自动绑定到本地地址设置为 INADDR_ANY 的随机空闲端口。

当在未绑定的套接字上调用connect(2) 时,该套接字将自动绑定到随机空闲端口或本地地址设置为 INADDR_ANY 的可用共享端口...

有几个特殊地址: INADDR_LOOPBACK (127.0.0.1) 总是通过环回设备引用本地主机; INADDR_ANY (0.0.0.0) 表示用于绑定的任何地址...

还:

bind() — 将名称绑定到套接字

如果 (sin_addr.s_addr) 字段设置为常量 INADDR_ANY,如 netinet/in.h 中所定义,则调用者请求将套接字绑定到主机上的所有网络接口。 随后,来自所有接口(与绑定名称匹配)的 UDP 数据包和 TCP 连接被路由到应用程序。 当服务器向多个网络提供服务时,这变得很重要。 通过不指定地址,服务器可以接受为其端口发出的所有 UDP 数据包和 TCP 连接请求,而不管请求到达的网络接口。

当您不需要将套接字绑定到特定 IP 时使用INADDR_ANY 当您在调用bind()时将此值用作地址时,套接字将接受与机器所有 IP 的连接。

要将套接字与localhost绑定,在调用绑定函数之前,应正确设置 sockaddr_in 结构的 sin_addr.s_addr 字段。 可以通过以下方式获得适当的值

my_sockaddress.sin_addr.s_addr = inet_addr("127.0.0.1")

或由

my_sockaddress.sin_addr.s_addr=htonl(INADDR_LOOPBACK);

INADDR_ANY指示侦听套接字绑定到所有可用接口。 这与尝试绑定到inet_addr("0.0.0.0") 为了完整起见,我还将提到 IPv6 也有IN6ADDR_ANY_INIT ,这与尝试绑定到 IPv6 套接字的::地址相同。

#include <netinet/in.h>

struct in6_addr addr = IN6ADDR_ANY_INIT;

另请注意,当您将 IPv6 套接字绑定到IN6ADDR_ANY_INIT您的套接字将绑定到所有 IPv6 接口,并且也应该能够接受来自 IPv4 客户端的连接(尽管 IPv6 映射地址)。

INADDR_ANY 是一个常量,在 value 中包含 0。 仅当您想从不关心 ip-add 的所有活动端口进行连接时才使用。 所以如果你想连接任何特定的 ip 你应该提到像 my_sockaddress.sin_addr.s_addr = inet_addr("192.168.78.2")

当你有一个服务器时,你创建一个套接字,然后将它绑定到一个 IP 和端口,这为套接字提供了一种基于唯一的套接字类型、地址族、IP 和端口进行识别的唯一方式。 然后使用listen() 将套接字设置为服务器模式,然后执行accept(),它等待一个连接,该连接将具有目标参数的入站数据包,这些数据包会在该套接字上排队。

当你有一个客户端时,你创建一个套接字,然后将套接字连接()到远程 IP 和端口,如果它还没有被绑定,它也会将 0.0.0.0 和一个随机未使用的临时端口绑定到套接字使用 bind(INADDR_ANY, 0) 到 IP 和端口。 connect() 连接时返回,并使用 IP 和端口作为出站数据包中的源地址,其中 0.0.0.0 始终由操作系统替换为当前内部 IP,然后您使用 sendall 发送应用程序数据。

INADDR_ANY 比以编程方式获取计算机当前的内部 IP 更快,该 IP 可能随时更改,并且端口上将不再接收数据包,但它们仍会在 0.0.0.0 上接收,因为它是任何地址。

请注意,套接字可以绑定到 0.0.0.0,但不能绑定到端口 0,因为它是一个通配符,可以为套接字提供一个随机的临时端口,因此当您使用 bind(INADDR_ANY, 0) 时,它绑定到 0.0.0.0 和一个随机临时端口。

#include <arpa/inet.h>                                                                     
.                                                                                   
.                                                                                        
tcpsock.sin_addr.s_addr = inet_addr("192.168.1.2")

为我工作

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM