您的位置:首页 > 理论基础 > 计算机网络

UDP 和 TCP 对比讲解

2016-03-28 23:03 381 查看
在.Net中,System.Net.Sockets命名空间为需要严密控制网络访问的开发人员提供了WindowsSockets(Winsock)接口的托管实现。System.Net命名空间中的所有其他网络访问类都建立在该套接字Socket实现之上,如TCPClient、TCPListener和UDPClient类封装有关创建到Internet的TCP和UDP连接的详细信息;NetworkStream类则提供用于网络访问的基础数据流等,常见的许多Internet服务都可以见到Socket的踪影,如Telnet、Http、Email、Echo等,这些服务尽管通讯协议Protocol的定义不同,但是其基础的传输都是采用的Socket。

其实,Socket可以象流Stream一样被视为一个数据通道,这个通道架设在应用程序端(客户端)和远程服务器端之间,而后,数据的读取(接收)和写入(发送)均针对这个通道来进行。

可见,在应用程序端或者服务器端创建了Socket对象之后,就可以使用Send/SentTo方法将数据发送到连接的Socket,或者使用Receive/ReceiveFrom方法接收来自连接Socket的数据;

针对Socket编程,.NET框架的Socket类是Winsock32API提供的套接字服务的托管代码版本。其中为实现网络编程提供了大量的方法,大多数情况下,Socket类方法只是将数据封送到它们的本机Win32副本中并处理任何必要的安全检查。如果你熟悉WinsockAPI函数,那么用Socket类编写网络程序会非常容易,当然,如果你不曾接触过,也不会太困难,跟随下面的解说,你会发觉使用Socket类开发windows网络应用程序原来有规可寻,它们在大多数情况下遵循大致相同的步骤。

在使用之前,你需要首先创建Socket对象的实例,这可以通过Socket类的构造方法来实现:

publicSocket(AddressFamilyaddressFamily,SocketTypesocketType,ProtocolTypeprotocolType);

其中,addressFamily参数指定Socket使用的寻址方案,socketType参数指定Socket的类型,protocolType参数指定Socket使用的协议。

下面的示例语句创建一个Socket,它可用于在基于TCP/IP的网络(如Internet)上通讯。

Sockets=newSocket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

若要使用UDP而不是TCP,需要更改协议类型,如下面的示例所示:

Sockets=newSocket(AddressFamily.InterNetwork,SocketType.Dgram,ProtocolType.Udp);

一旦创建Socket,在客户端,你将可以通过Connect方法连接到指定的服务器,并通过Send/SendTo方法向远程服务器发送数据,而后可以通过Receive/ReceiveFrom从服务端接收数据;而在服务器端,你需要使用Bind方法绑定所指定的接口使Socket与一个本地终结点相联,并通过Listen方法侦听该接口上的请求,当侦听到用户端的连接时,调用Accept完成连接的操作,创建新的Socket以处理传入的连接请求。使用完Socket后,记住使用Shutdown方法禁用Socket,并使用
Close方法关闭Socket。其间用到的方法/函数有:

Socket.Connect方法:建立到远程设备的连接

publicvoidConnect(EndPointremoteEP)(有重载方法)

Socket.Send方法:从数据中的指示位置开始将数据发送到连接的Socket。

publicintSend(byte[],int,SocketFlags);(有重载方法)

Socket.SendTo方法将数据发送到特定终结点。

publicintSendTo(byte[],EndPoint);(有重载方法)

Socket.Receive方法:将数据从连接的Socket接收到接收缓冲区的特定位置。

publicintReceive(byte[],int,SocketFlags);

Socket.ReceiveFrom方法:接收数据缓冲区中特定位置的数据并存储终结点。

publicintReceiveFrom(byte[],int,SocketFlags,refEndPoint);

Socket.Bind方法:使Socket与一个本地终结点相关联:

publicvoidBind(EndPointlocalEP);

Socket.Listen方法:将Socket置于侦听状态。

publicvoidListen(intbacklog);

Socket.Accept方法:创建新的Socket以处理传入的连接请求。

publicSocketAccept();

Socket.Shutdown方法:禁用某Socket上的发送和接收

publicvoidShutdown(SocketShutdownhow);

Socket.Close方法:强制Socket连接关闭

publicvoidClose();

可以看出,以上许多方法包含EndPoint类型的参数,在Internet中,TCP/IP使用一个网络地址和一个服务端口号来唯一标识设备。网络地址标识网络上的特定设备;端口号标识要连接到的该设备上的特定服务。网络地址和服务端口的组合称为终结点,在.NET框架中正是由EndPoint类表示这个终结点,它提供表示网络资源或服务的抽象,用以标志网络地址等信息。.Net同时也为每个受支持的地址族定义了EndPoint的子代;对于IP地址族,该类为IPEndPoint。IPEndPoint类包含应用程序连接到主机上的服务所需的主机和端口信息,通过组合服务的主机IP地址和端口号,IPEndPoint
类形成到服务的连接点。

用到IPEndPoint类的时候就不可避免地涉及到计算机IP地址,.Net中有两种类可以得到IP地址实例:

IPAddress类:IPAddress类包含计算机在IP网络上的地址。其Parse方法可将IP地址字符串转换为IPAddress实例。下面的语句创建一个IPAddress实例:

IPAddressmyIP=IPAddress.Parse("192.168.1.2");

Dns类:向使用TCP/IPInternet服务的应用程序提供域名服务。其Resolve方法查询DNS服务器以将用户友好的域名(如"host.contoso.com")映射到数字形式的Internet地址(如192.168.1.1)。Resolve方法返回一个IPHostEnty实例,该实例包含所请求名称的地址和别名的列表。大多数情况下,可以使用AddressList数组中返回的第一个地址。下面的代码获取一个IPAddress实例,该实例包含服务器host.contoso.com
的IP地址。

IPHostEntryipHostInfo=Dns.Resolve("host.contoso.com");

IPAddressipAddress=ipHostInfo.AddressList[0];

你也可以使用GetHostName方法得到IPHostEntry实例:

IPHosntEntryhostInfo=Dns.GetHostByName("host.contoso.com")

在使用以上方法时,你将可能需要处理以下几种异常:

SocketException异常:访问Socket时操作系统发生错误引发

ArgumentNullException异常:参数为空引用引发

ObjectDisposedException异常:Socket已经关闭引发

在掌握上面得知识后,下面的代码将该服务器主机(host.contoso.com的IP地址与端口号组合,以便为连接创建远程终结点:

IPEndPointipe=newIPEndPoint(ipAddress,11000);

确定了远程设备的地址并选择了用于连接的端口后,应用程序可以尝试建立与远程设备的连接。下面的示例使用现有的IPEndPoint实例与远程设备连接,并捕获可能引发的异常:

try{

s.Connect(ipe);//尝试连接

}

//处理参数为空引用异常

catch(ArgumentNullExceptionae){

Console.WriteLine("ArgumentNullException:{0}",ae.ToString());

}

//处理操作系统异常

catch(SocketExceptionse){

Console.WriteLine("SocketException:{0}",se.ToString());

}

catch(Exceptione){

Console.WriteLine("Unexpectedexception:{0}",e.ToString());

}

需要知道的是:Socket类支持两种基本模式:同步和异步。其区别在于:在同步模式中,对执行网络操作的函数(如Send和Receive)的调用一直等到操作完成后才将控制返回给调用程序。在异步模式中,这些调用立即返回。

另外,很多时候,Socket编程视情况不同需要在客户端和服务器端分别予以实现,在客户端编制应用程序向服务端指定端口发送请求,同时编制服务端应用程序处理该请求,这个过程在上面的阐述中已经提及;当然,并非所有的Socket编程都需要你严格编写这两端程序;视应用情况不同,你可以在客户端构造出请求字符串,服务器相应端口捕获这个请求,交由其公用服务程序进行处理。以下事例语句中的字符串就向远程主机提出页面请求:

stringGet="GET/HTTP/1.1\r\nHost:"+server+"\r\nConnection:Close\r\n\r\n";

远程主机指定端口接受到这一请求后,就可利用其公用服务程序进行处理而不需要另行编制服务器端应用程序。

综合运用以上阐述的使用VisualC#进行Socket网络程序开发的知识,下面的程序段完整地实现了Web页面下载功能。用户只需在窗体上输入远程主机名(Dns主机名或以点分隔的四部分表示法格式的IP地址)和预保存的本地文件名,并利用专门提供Http服务的80端口,就可以获取远程主机页面并保存在本地机指定文件中。如果保存格式是.htm格式,你就可以在Internet浏览器中打开该页面。适当添加代码,你甚至可以实现一个简单的浏览器程序。

实现此功能的主要源代码如下:

C#code
//"开始"按钮事件privatevoidbutton1_Click(objectsender,System.EventArgse){//取得预保存的文件名stringfileName=textBox3.Text.Trim();//远程主机stringhostName=textBox1.Text.Trim();//端口intport=Int32.Parse(textBox2.Text.Trim());//得到主机信息IPHostEntryipInfo=Dns.GetHostByName(hostName);//取得IPAddress[]IPAddress[]
ipAddr=ipInfo.AddressList;//得到ipIPAddressip=ipAddr[0];//组合出远程终结点IPEndPointhostEP=newIPEndPoint(ip,port);//创建Socket实例Socketsocket=newSocket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);try{//尝试连接socket.Connect(hostEP);}catch(Exception
se){MessageBox.Show("连接错误"+se.Message,"提示信息,MessageBoxButtons.RetryCancel,MessageBoxIcon.Information);}//发送给远程主机的请求内容串stringsendStr="GET/HTTP/1.1\r\nHost:"+hostName+"\r\nConnection:Close\r\n\r\n";//创建bytes字节数组以转换发送串byte[]bytesSendStr=newbyte[1024];//将发送内容字符串转换成字节byte数组bytesSendStr=Encoding.ASCII.GetBytes(sendStr);try{//向主机发送请求socket.Send(bytesSendStr,bytesSendStr.Length,0);
}catch(Exceptionce){MessageBox.Show("发送错误:"+ce.Message,"提示信息,MessageBoxButtons.RetryCancel,MessageBoxIcon.Information);}//声明接收返回内容的字符串stringrecvStr="";//声明字节数组,一次接收数据的长度为1024字节byte[]recvBytes=newbyte[1024];//返回实际接收内容的字节数intbytes=0;//循环读取,直到接收完所有数据while(true)
{bytes=socket.Receive(recvBytes,recvBytes.Length,0);//读取完成后退出循环if(bytes<=0)break;//将读取的字节数转换为字符串recvStr+=Encoding.ASCII.GetString(recvBytes,0,bytes);}//将所读取的字符串转换为字节数组byte[]content=Encoding.ASCII.GetBytes(recvStr);try{//创建文件流对象实例FileStreamfs=newFileStream(fileName,FileMode.OpenOrCreate,FileAccess.ReadWrite);//写入文件fs.Write(content,0,content.Length);
}catch(Exceptionfe){MessageBox.Show("文件创建/写入错误:"+fe.Message,"提示信息",MessageBoxButtons.RetryCancel,MessageBoxIcon.Information);}//禁用Socketsocket.Shutdown(SocketShutdown.Both);//关闭Socketsocket.Close();}}


程序在WindowsXP中文版、.NetFrameworkd中文正式版、VisualStudio.Net中文正式版下调试通过

////////////////////////////////////////

对于TCP的Socket编程,主要分二部分:

一、服务端Socket侦听:

服务端Socket侦听主要分以下几个步骤,按照以下几个步骤我们可以很方便的建立起一个Socket侦听服务,来侦听尝试连接到该服务器的客户Socket,从而建立起连接进行相关通讯。

1、创建IPEndPoint实例,用于Socket侦听时绑定

1

IPEndPointipep=newIPEndPoint(IPAddress.Any,6001);

2、创建套接字实例

1

//创建一个套接字

2

serverSocket=newSocket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

这里创建的时候用ProtocolType.Tcp,表示建立一个面向连接(TCP)的Socket。

3、将所创建的套接字与IPEndPoint绑定

1

//将所创建的套接字与IPEndPoint绑定

2

serverSocket.Bind(ipep);

4、设置套接字为收听模式

1

//设置套接字为收听模式

2

serverSocket.Listen(10);

以上这四步,我们已经建立了Socket的侦听模式,下面我们就来设置怎么样来获取客户Socket连接的实例,以及连接后的信息发送。

5、在套接字上接收接入的连接

1

while(true)

2

{

3

try

4

{

5

//在套接字上接收接入的连接

6

clientSocket=serverSocket.Accept();

7

clientThread=newThread(newThreadStart(ReceiveData));

8

clientThread.Start();

9

}

10

catch(Exceptionex)

11

{

12

MessageBox.Show("listeningError:"+ex.Message);

13

}

14

}

通过serverSocket.Accept()来接收客户Socket的连接请求,在这里用循环可以实现该线程实时侦听,而不是只侦听一次。当程序运行serverSocket.Accept()时,会等待,直到有客户端Socket发起连接请求时,获取该客户Socket,如上面的clientSocket。在这里我用多线程来实现与多个客户端Socket的连接和通信,一旦接收到一个连接后,就新建一个线程,执行ReceiveData功能来实现信息的发送和接收。

6、在套接字上接收客户端发送的信息和发送信息

1

privatevoidReceiveData()

2

{

3

boolkeepalive=true;

4

Sockets=clientSocket;

5

Byte[]buffer=newByte[1024];

6


7

//根据收听到的客户端套接字向客户端发送信息

8

IPEndPointclientep=(IPEndPoint)s.RemoteEndPoint;

9

lstServer.Items.Add("Client:"+clientep.Address+"("+clientep.Port+")");

10

stringwelcome="Welcometomytestsever";

11

byte[]data=newbyte[1024];

12

data=Encoding.ASCII.GetBytes(welcome);

13

s.Send(data,data.Length,SocketFlags.None);

14


15

while(keepalive)

16

{

17

//在套接字上接收客户端发送的信息

18

intbufLen=0;

19

try

20

{

21

bufLen=s.Available;

22


23

s.Receive(buffer,0,bufLen,SocketFlags.None);

24

if(bufLen==0)

25

continue;

26

}

27

catch(Exceptionex)

28

{

29

MessageBox.Show("ReceiveError:"+ex.Message);

30

return;

31

}

32

clientep=(IPEndPoint)s.RemoteEndPoint;

33

stringclientcommand=System.Text.Encoding.ASCII.GetString(buffer).Substring(0,bufLen);

34


35

lstServer.Items.Add(clientcommand+"("+clientep.Address+":"+clientep.Port+")");

36


37

}

38


39

}

通过IPEndPointclientep=(IPEndPoint)s.RemoteEndPoint;我们可以获取连接上的远程主机的端口和IP地址,如果想查询该主机的其它属性如主机名等,可用于上一篇讲的Dns.GetHostByAddress(stringipAddress)来返回一个IPHostEntry对象,就可以得到。另外我们要注意的是,通过Socket发送信息,必须要先把发送的信息转化成二进字进行传输,收到信息后也要把收到的二进字信息转化成字符形式,这里可以通过Encoding.ASCII.GetBytes(welcome);和Encoding.ASCII.GetString(buffer).Substring(0,
bufLen);来实现。

以上就是服务端Socket侦听模式的实现,只要有远程客户端Socket连接上后,就可以轻松的发送信息和接收信息了。下面我们来看看客户端Socket是怎么连接上服务器的。

二、客户端连接

客户端Socket连接相对来说比较简单了,另外说明一下,在执行客户端连接前,服务端Socket侦听必须先启动,不然会提示服务器拒绝连接的信息。

1、创建IPEndPoint实例和套接字

1

//创建一个套接字

2

IPEndPointipep=newIPEndPoint(IPAddress.Parse("127.0.0.1"),6001);

3

clientSocket=newSocket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);

这个跟服务端Socket侦听差不多,下面一步由服务端Socket的侦听模式变成连接模式。

2、将套接字连接到远程服务器

1

//将套接字与远程服务器地址相连

2

try

3

{

4

clientSocket.Connect(ipep);

5

}

6

catch(SocketExceptionex)

7

{

8

MessageBox.Show("connecterror:"+ex.Message);

9

return;

10

}

前面已说明,如果在执行Socket连接时,服务器的Socket侦听没有开启的话,会产生一个SocketException异常,如果没有异常发生,那恭喜你,你已经与服务器连接上了,接下来就可以跟服务器通信了。

3、接收信息

1

while(true)

2

{

3

//接收服务器信息

4

intbufLen=0;

5

try

6

{

7

bufLen=clientSocket.Available;

8


9

clientSocket.Receive(data,0,bufLen,SocketFlags.None);

10

if(bufLen==0)

11

{

12

continue;

13

}

14

}

15

catch(Exceptionex)

16

{

17

MessageBox.Show("ReceiveError:"+ex.Message);

18

return;

19

}

20


21

stringclientcommand=System.Text.Encoding.ASCII.GetString(data).Substring(0,bufLen);

22


23

lstClient.Items.Add(clientcommand);

24


25

}

4、发送信息

1

//向服务器发送信息

2


3

byte[]data=newbyte[1024];

4

data=Encoding.ASCII.GetBytes(txtClient.Text);

5

clientSocket.Send(data,data.Length,SocketFlags.None);

客户端的发送信息和接收信息跟服务器的接收发送是一样的,只不过一个是侦听模式而另一个是连接模式。

以下是程序的运行界面,这些在源码下载里都可以看到:

1、服务端界面:



2、客户端界面:



好了,关于面向连接的Socket就讲到这里了,以实例为主,希望对那些派得上用场的朋友能够看得明白。另外提一下,这里服务端开启侦听服务、客户端连接服务端都采用线程方式来实现,这样服务端能够跟多个客户端同时通信,不用等候,当然还有另外一种方式可以实现那就是异步socket,关于这些可以搜索博客园上的相关文章,已经有好多了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: