1. 通过bind机制, socket必须要调用bind才能发送tcp包。 bind调用时需要一个ip地址。一般一台机器的多网口都要配置不同的ip地址(路由器除外,路由器是一个网桥设备,只是负责转发包,所以其它的端口一般没有ip地址)。
2. 通过ioctl来设置打开的socket. (ioctl (fd, SIOCGIFINDEX, &ifr)==0;
TCP编程时不管是客户端还是服务器端,都要调用bind后才能连接/收发数据。
UDP在客户端时可以不调用bind而直接使用recvfrom/sendto来收发数据。
在客户端一般是本地地址选择INETADDR_ANY, 表示所有网络接口。
代码如下:
#include ; #include ; #include ; #include ; #include ; #include ; #include ; #include ; int main () { struct sockaddr_ll sll; int fd; struct ifreq ifr; char *dev; fd = socket (PF_PACKET, SOCK_RAW, htons(ETH_P_ALL)); dev = "eth0"; strncpy ((char *)ifr.ifr_name, dev, sizeof(ifr.ifr_name)); assert (ioctl (fd, SIOCGIFINDEX, &ifr)==0); memset (&sll, 0, sizeof(sll)); sll.sll_family = AF_PACKET; sll.sll_protocol = htons (ETH_P_ALL); sll.sll_ifindex = ifr.ifr_ifindex; assert (bind (fd, (struct sockaddr *)&sll, sizeof(sll))==0); }
##############################################################
Windows网络编程总结(一) 作者: Kendiv 出处: CSDN.NET 关于bind: INADDR_ANY 的具体含义是,绑定到0.0.0.0。此时,对所有的地址都将是有效的,如果系统考虑冗余,采用多个网卡的话,那么使用此种bind,将在所有网卡上进行绑定。在这种情况下,你可以收到发送到所有有效地址上数据包。 例如: SOCKADDR_IN Local; Local.sin_addr.s_addr = htonl(INADDR_ANY); 另外一种方式如下: SOCKADDR_IN Local; hostent* thisHost = gethostbyname( " "); char* ip = inet_ntoa(*(struct in_addr *)*thisHost-> h_addr_list); Local.sin_addr.s_addr = inet_addr(ip); 在这种方式下,将在系统中当前第一个可用地址上进行绑定。在多网卡的环境下,可能会出问题。 最常见的方式: const char LocalIP[] = "192.168.0.100 "; SOCKADDR_IN Local; Local.sin_addr.s_addr = inet_addr(LocalIP); bind(socket, (LPSOCKADDR)&Local, sizeof(SOCKADDR_IN) bind的安全问题: 如果你在bind时,使用了INADDR_ANY那么,你将可以在所有有效的地址上进行监听,但是Socket有一个特性:可在同一端口上绑定多个Socket。 让我们看看下面的情况:假设你的系统只有一个IP:192.168.0.100,你希望bind到4096端口。对于下面的两种bind,当数据包到达时,谁会接收到呢? Local.sin_addr.s_addr = htonl(INADDR_ANY); Local.sin_addr.s_addr = inet_addr(“192.168.0.100”); WinSocke库是这样处理的:谁绑定的最明确,谁获取数据包。显然,第二种bind将获取到达的数据包。如果避免这种情况呢?使用SO_EXECLUSINEADDRUSE选项。需要注意的是,此选项在Windows NT 4 Service Pack 4以后(包括SP4)才可以使用。 示例代码: #ifndef SO_EXECLUSINEADDRUSE #define SO_EXECLUSINEADDRUSE ((int)(~SO_REUSEADDR)) #endif SOCKADDR_IN Local; BOOL val = TRUE; Local. sin_family = AF_INET; Local. sin_port = htons(4096); Local.sin_addr.s_addr = htonl(INADDR_ANY); setsocketopt(socket, SOL_SOCKET, SO_EXECLUSINEADDRUSE, (char*)&val, sizeof(val)); bind(socket, (LPSOCKADDR)&Local, sizeof(SOCKADDR_IN)