基于TCP协议的异步通讯(c#)
1.建立连接:
在同步模式中,我们在服务器上使用Accept方法接入连接请求,而 在客户端我们则使用Connect方法来连接服务器。相对地,在异步模式下,服务器可以使用BeginAccept方法和EndAccept方法来完成连接到客户端的任务,在客户端则通过BeginConnect方法和EndConnect方法来实现与服务器的连接。
BeginAccept在异步方式下传入的连接尝试,它允许其他动作而不必等待连接建立才继续执行后面程序在调用BeginAccept之前,必须使用Listen方法来侦听是否有连接请求,等待他把请求放入消息队列中参数列表为:
BeginAccept(AsyncCallBack, Ojbect state)
AsyncCallBack:代表回调函数
state:表示状态信息,必须保证state中包含socket的句柄
而在回调方法中应调用EndAccept()方法来完成操作,该方法返回新的套接字对象,并在以后的通信中可以使用该套接字。其基本流程是:
(1)创建本地终节点,并新建套接字与本地终节点进行绑定;
(2)在端口上侦听是否有新的连接请求;
(3)请求开始接入新的连接,传入Socket的实例或者StateOjbect的实例
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
IPAddress local=IPAddress.Prase("127.0,0,1");
IPEndPoint iep=new IPEndPoint(local,13000);
//创建服务器的socket对象
server=new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp)
server.Bind(iep);
server.Listen(20);
Server.BeginAccecpt(New AsyncCallBack(Accept),server);
当BeginAccept()方法调用结束后,一旦新的连接发生,将调用回调函数,而该回调函数必须包括用来结束接入连接操作的EndAccept()方法。
该方参数列表为 Socket EndAccept(IAsyncResult iar)
下面为回调函数的实例:
{
//还原传入的原始套接字
Socket MyServer=(Socket)iar.ASyncState;
//在原始套接字上调用EndAccept方法,返回新的套接字
service=MyServer.EndAccept(iar);
}
服务器端已经准备好了,那么客户端应通过BeginConnect方法和EndConnect来远程连接主机。在调用BeginConnect方法时必须注册相应的回调函数并且至少传第一个Socket的实例给state参数,以保证EndConnect方法中能使用原始的套接字。下面是一段BeginConnect的调用:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
IPAddress ip=IPAddress.Prase("127.0.0.1");
IpEndPoint iep=new IPEndpoint(ip,13000);
sock.BeginConnect(iep, new AsyncCallBack(Connect),sock);
EndConnect是一种阻塞方法,用于完成BeginConnect方法的异步连接诶远程主机的请求。在注册了回调函数后必须接收BeginConnect方法返回的IASynccReuslt作为参数。下面为代码演示:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
{
client=(socket)iar.AsyncStete;
try
{
client.EndConnect(iar);
}
Catch(SocketException)
{
Console.WriteLine("无法连接");
}
Finally
{
}
}
下面对关于TcpListener类使用BeginAccetpTcpClient方法处理一个传入的连接尝试。下面是使用BeginAccetpTcpClient方法的代码:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
{
//开始从客户端监听连接
Console.WriteLine("Waitting for a connection");
//接收连接
//开始准备接入新的连接,一旦有新连接尝试则调用回调函数DoAcceptTcpCliet
listner.BeginAcceptTcpClient
(
new AsyncCallBack(DoAcceptTcpCliet),
listner
);
}
//处理客户端的连接
public static void DoAcceptTcpCliet(IAsyncResult iar)
{
//还原原始的TcpListner对象
TcpListner listener=(TcpListner)iar.AsyncState;
//完成连接的动作,并返回新的TcpClient
TcpClient client=listner.EndAcceptTcp(iar);
Console.WriteLine("连接成功");
}
代码的处理业务逻辑为:
(1)调用BeginAccetpTcpClient方法开开始连接新的连接,当连接视图发生时,回调函数被调用以完成
连接操作;
(2)上面DoAcceptTcpCliet方法通过AsyncState属性获得由BeginAcceptTcpClient传入的listner实例
(3)在得到listner对象后,用它调用EndAcceptTcpClient方法,该方法返回新的包含客户端信息的TcpClient。
BeginConnect方法和EndConnect方法可用于客户端建立与服务端的连接,下面看实例:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
{
//开始与远程主机进行连接
client.BeginConnect(serverIP[0],13000,requestCallBack,client);
Console.WriteLine("开始与服务器进行连接");
}
Priveate void requestCallBack(IAsyncResult iar)
{
try
{
//还原原始的TcpClient对象
client=(TcpClient)iar.AsyncState;
//
client.EndConnect(iar);
Console.WriteLine("与服务器{0}连接成功",client.Client.RemoteEndPoint);
}
Catch(Exception e)
{
Console.WriteLine(e.toString());
}
Finally
{
}
}
2.发送与接受数据:
在建立了套接字的连接后,就可以服务器端和客户端之间进行数据通信了。异步套接字用BeginSend和EndSend方法来负责数据的发送。注意在调用BeginSend方法前要确保双方都已经建立连接,否则会出异常。下面演示代码:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
{
try
{
//
Socket socket=(Scoket)iar.AsyncState;
socket=.EndConnect(iar);
byte[] buff=Encoding.ASCII.GetBytes("This is a test!");
s.BeginSend(buff,0,buff.Length,new AsycCallBack(SendCallBack),socket=);
}
Catch(Exception e)
{
Console.WriteLine(e.toString());
}
Finally
{
}
}
//回调函数
Public void SendCallBack(IAsyncResult iar)
{
try
{
Socket s=(Socket)iar.IAsyncState;
int send=s.EndSend(iar);
}
Catch(Exception)
{
Console.WriteLine(e.toString());
}
Finally
{
}
}
服务器接收数据是通过BeginReceive和EndReceive方法:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
public static void Listen_CallBack(IAsyncResult iar)
{
Socket s=(Socket)iar.IAsyncState;
Socket s2=s.EndAccept(iar);
s2.BeginReceive(buffer,0,buffer.length, new AsyncCallBack(Read_Callback),s2);
}
public static void Read_Callback(IAsyncResult iar)
{
Socket s=(Socket)iar.IAsyncState;
int recv=s.EndReceive(iar);
string str=Encoding.ASCII.GetString(buffer,0,recv);
Console.WriteLine(str);
}
(1)首先处理连接的回调函数里得到的通讯套接字s2,接着开始接收数据
(2)当数据发送到缓冲区中,BeginReceive方法试图从buffer数组中读取长度为buffer.length的数据块,并返回接收到的数据量recv。最后接收并打印数据。
下面是基于NetworkStream相关的异步发送和接收方法。
NetworkStream使用BeginRead和EndRead方法进行读操作,使用BeginWreite和EndWrete方法进行写操作,下面看实例:
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
{
TcpClient tcpClient=client;
//使用TcpClient的GetStream方法获取网络流
NetworkStream ns=tcpClient.GetStrem();
//检查网络流是否可读
if(ns.CanRead)
{
//定义缓冲区
byte[] read=new byte[1024];
ns.BeginRead(read,0,read.Length,new AsyncCallBack(myReadCallBack),ns);
}
Else
{
Console.WriteLine("无法从网络中读取流数据");
}
}
public static void myReadCallBack(IAsyncResult iar)
{
NetwrokStream ns=(NetwrokStream)iar.AsyncState;
byte[] read=new byte[1024];
String data="";
int recv;
recv=ns.EndReaD(iar);
data=String.Concat(data,Encoding.ASCII.GetString(read,0,recv));
//接收到的消息长度可能大于缓冲区总大小,反复循环直到读完为止
while(ns.DataAvailable)
{
ns.BeginRead(read,0,read.Length, new AsyncCallBack(myReadCallBack),ns);
}
//打印
Console.WriteLine("您收到的信息是"+data);
}
3.程序阻塞与异步中的同步问题:
.Net里提供了EventWaitHandle类来表示一个线程的同步事件。
EventWaitHandle即事件等待句柄,他允许线程通过操作系统互发信号和等待彼此的信号来达到线程同步的目的。
这个类有2个子类,分别为AutoRestEevnt(自动重置)和ManualRestEvent(手动重置)。下面是线程同步的几个方法:
(1)Rset方法:将事件状态设为非终止状态,导致线程阻塞。这里的线程阻塞是指允许其他需要等待的线程进行阻塞
即让含WaitOne()方法的线程阻塞;
(2)Set方法:将事件状态设为终止状态,允许一个或多个等待线程继续。该方法发送一个信号给操作系统,让处于等待的某个线程从阻塞状态转换为继续运行,即WaitOne方法的线程不在阻塞;
(3)WaitOne方法:阻塞当前线程,直到当前的等待句柄收到信号。此方法将一直使本线程处于阻塞状态直到收到信号为止,
即当其他非阻塞进程调用set方法时可以继续执行。
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
IpEndPoint iep=new IpEndPoint(ipHostEntry.AddressList[0],13000);
Socket s=new Socket(iep.Address.AddressFamily,SocketType.Stream,ProtocalType.Tcp);
//设置标志位
private ManualResetEvent allDone=new ManualResetEvent(false);
try
{
//绑定并监听
s.Bind(iep);
s.Listen(13000);
while(true)
{
//让其他需要等待的线程阻塞
allDone.Reset();
Console.WriteLine("Waitting for a connect");
s.BeginAccept(new AsyncCallBack(Listen_Callback),s);
//阻塞当前进程直到接收到信号
allDone.WaitOne();
}
}
catach(Exception e)
{
Conso.WriteLine(e.toString());
}
Finally
{
}
public static void Listen_Callback(IAsyncResult iar)
{
//发送信号让阻塞的线程继续执行
allDone.Set();
Socket s=(Socket)iar.AsyncState;
Socket ns=s.EndAccept(iar);
}
(1)试用了ManualRestEvent对象创建一个等待句柄,在调用BeginAccept方法前使用Rest方法允许其他线程阻塞;
(2)为了防止在连接完成之前对套接字进行读写操作,务必要在BeginAccept方法后调用WaitOne来让线程进入阻塞状态。
当有连接接入后系统会自动调用会调用回调函数,所以当代码执行到回调函数时说明连接已经成功,并在函数的第一句
就调用Set方法让处于等待的线程可以继续执行。