linux – 套接字级别的IPv4和IPv6之间有什么区别吗?

前端之家收集整理的这篇文章主要介绍了linux – 套接字级别的IPv4和IPv6之间有什么区别吗?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我需要为现有的仅支持IPv4的基于套接字的应用程序添加IPv6支持.为了迎接即将到来的IPv6时代,有人命令我为应用程序添加IPv6接口,并在尝试与应用程序通信时让外界选择IPv4接口或IPv6接口.

我的问题是:对于Linux中的套接字处理API级别,处理基于IPv4的套接字和基于IPv6的套接字之间没有区别吗?

更多的是,是否可以让套接字侦听具有相同端口的两个IP地址?如果这是真的,那么实现这个要求是一项微不足道的工作,我想.

解决方法

用1个TCP套接字监听2个不同的IP地址是不可能的,但是如果你使用 in6addr_any地址监听所有接口,那么也将包括所有的IPv4地址(虽然我相信例如linux有一个内核选项来禁用该映射).

对于您是使用IPv4还是IPv6,(较新版本)套接字API非常透明,但必须非常谨慎地处理IPv4应用程序的编码方式.

例如此IPv4代码接受连接并打印出远程主机的地址:

struct sockaddr_in client_addr;
 socklen_t addr_len = sizeof(client_addr);
 client_data->fd = accept(server_fd,(struct sockaddr*)&client_addr,&addr_len);
 log_printf("New client from %s\n",inet_ntoa(client_addr.sin_addr.s_addr));

必须转换为以下内容,它处理IPv4和IPv6

struct sockaddr_storage client_addr;
 char numeric_addr[INET6_ADDRSTRLEN];
 socklen_t addr_len = sizeof(client_addr);
 client_data->fd = accept(server_fd,&addr_len);
 if(client_addr.ss_family == AF_INET)
    log_printf("New client from %s\n",inet_ntop(client_addr.ss_family,((struct sockaddr_in*)&client_addr)->sin_addr.s_addr,numeric_addr,sizeof numeric_addr));
 else if(client_addr.ss_family == AF_INET6)
    log_printf("New client from %s\n",((struct sockaddr_in6*)&client_addr)->sin6_addr,sizeof numeric_addr));

虽然我相信你可以用getaddrinfo()更优雅和透明地做到这一点

以下是有关IP层独立性的附加说明:
http://uw714doc.sco.com/en/SDK_netapi/sockC.PortIPv4appIPv6.html
http://www.kame.net/newsletter/19980604/

猜你在找的Linux相关文章