转自:https://www.cnblogs.com/zhuweisky/archive/2012/07/18/2597613.html 
作者:zhuweisky 


      随着互联网越来越普及,以及物联网的兴起,IPv4地址已远远不够用,IPv6的普及将是不可避免的趋势。以前,我们的大部分socket程序几乎都是针对IPv4而开发,如果不做升级重构,那么使用IPv6地址的客户端将无法使用服务端提供的服务。如何才能像ESFramework一样,使服务端和客户端都可以同时支持IPv6了?使我们的P2P打洞也兼容IPv6了?下面我们将要点一一点出。

      首先,要了解两个最基础的事实:

(1)通信的双方,无论是服务端与客户端之间,或是客户端与客户端之间的P2P通信,必须使用相同的协议版本 -- 要么都是IPv4,要么都是IPv6。

(2)在没有特别安装附件的情况下,有的OS可能只支持IPv4,有的可能只支持IPv6,有的即支持IPv4也支持IPv6。可以通过Socket类的OSSupportsIPv6和OSSupportsIPv4属性来作判断。

一.TCP服务端     

      要让TCP服务端即能够接收IPv4地址的客户端的请求,也能接收IPv6地址客户端的请求,前提是服务器的OS即支持IPv4也支持IPv6。默认的,windows 2003 Server 是只支持IPv4的,可以通过安装协议来使其支持IPv6。

      然后,写服务端程序时,必需同时监听本机IPv4地址和IPv6地址,并且是监听这两个地址的同一个端口。比如,像下面这样:

  1. int port = 9900;
  2. TcpListener tcpListenerV4 = new TcpListener(IPAddress.Any, port);
  3. TcpListener tcpListenerV6 = new TcpListener(IPAddress.IPv6Any, port);

     如此,客户端无论是使用的IPv4还是IPv6,其向服务端发起连接请求时,都可以被服务端接受。

二.TCP客户端

      我们现在假设服务端程序已经兼容了IPv6,并且其提供服务的IPv4地址为192.168.0.104,IPv6地址为fe80::14d8:a209:89e6:c162%14。

      那么TCP客户端在与服务端建立连接之前,要看本地OS对IPv4和IPv6的支持情况:

(1)如果本地OS仅支持IPv4,或者同时支持IPv4和IPv6,那么简单地,就让其连接到服务器的IPv4地址。示例代码如下所示:

  1. TcpClient client = new TcpClient(AddressFamily.InterNetwork);
  2. client.Connect("192.168.0.104", 9900);
(2)如果本地OS仅支持IPv6,那么,就让其连接到服务器的IPv6地址。示例代码如下所示:

  1. TcpClient client = new TcpClient(AddressFamily.InterNetworkV6);
  2. client.Connect("fe80::14d8:a209:89e6:c162%14", 9900);

三.UDP 

      对于UDP而言,服务端和客户端可以采用完全一样的模型。要让基于UDP的应用程序兼容IPv6,会稍微复杂一些。

(1)需要创建两个UdpClient实例,一个用于IPv4,一个用于IPv6。示例代码如下所示:

  1. int port = 9800;
  2. UdpClient udpClient4 = new UdpClient(port, AddressFamily.InterNetwork);
  3. UdpClient udpClient6 = new UdpClient(port ,AddressFamily.InterNetworkV6);
(2)需要在两个UdpClient实例上调用接收数据的方法,来接收数据。
(3)发送数据时,需要根据目标地址是IPv4还是IPv6,来选择正确的UdpClient实例进行发送。示例代码如下所示:
  1. public void Send(byte[] data, IPEndPoint endPoint)
  2.     {
  3.         if (endPoint.AddressFamily == AddressFamily.InterNetwork)
  4.         {
  5.             this.udpClient4.Send(data, data.Length, endPoint);
  6.         }
  7.         else
  8.         {
  9.             this.udpClient6.Send(data, data.Length, endPoint);
  10.         }
  11.     }

      上面的示例,我们是假设当前的OS同时支持IPv4和IPv6,如果仅仅支持其中的一个,那么就应该只创建udpClient4或udpClient6一个实例。

四.P2P与IPv6

      如果我们的TCP客户端以及UDP都按照了上面类似的方式进行了重构升级,那么,无论是基于TCP的P2P打洞,还是基于UDP的P2P打洞,其逻辑代码都不需要做任何修改,就可以完全兼容IPv6了。 


      本文只是列出了将Socket应用程序重构升级使其支持IPv6的要点,在实际实现的过程中,还有很多的细节需要处理,才能在现实的复杂环境中正常运行。这里就不再赘述了,有疑问的朋友可以留言讨论。谢谢。


09-24 03:13