【转】C#高性能大容量SOCKET并发(二):SocketAsyncEventArgs封装
http://blog.csdn.net/sqldebug_fan/article/details/17557341
1、SocketAsyncEventArgs介绍
SocketAsyncEventArgs是微软提供的高性能异步Socket实现类,主要为高性能网络服务器应用程序而设计,主要是为了避免在在异步套接字 I/O 量非常大时发生重复的对象分配和同步。使用此类执行异步套接字操作的模式包含以下步骤:
1.分配一个新的 SocketAsyncEventArgs 上下文对象,或者从应用程序池中获取一个空闲的此类对象。
2.将该上下文对象的属性设置为要执行的操作(例如,完成回调方法、数据缓冲区、缓冲区偏移量以及要传输的最大数据量)。
3.调用适当的套接字方法 (xxxAsync) 以启动异步操作。
4.如果异步套接字方法 (xxxAsync) 返回 true,则在回调中查询上下文属性来获取完成状态。
5.如果异步套接字方法 (xxxAsync) 返回 false,则说明操作是同步完成的。 可以查询上下文属性来获取操作结果。
6.将该上下文重用于另一个操作,将它放回到应用程序池中,或者将它丢弃。
2、SocketAsyncEventArgs封装
使用SocketAsyncEventArgs之前需要先建立一个Socket监听对象,使用如下代码:
- public void Start(IPEndPoint localEndPoint)
- {
- listenSocket = new Socket(localEndPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
- listenSocket.Bind(localEndPoint);
- listenSocket.Listen(m_numConnections);
- Program.Logger.InfoFormat("Start listen socket {0} success", localEndPoint.ToString());
- //for (int i = 0; i < 64; i++) //不能循环投递多次AcceptAsync,会造成只接收8000连接后不接收连接了
- StartAccept(null);
- m_daemonThread = new DaemonThread(this);
- }
然后开始接受连接,SocketAsyncEventArgs有连接时会通过Completed事件通知外面,所以接受连接的代码如下:
- public void StartAccept(SocketAsyncEventArgs acceptEventArgs)
- {
- if (acceptEventArgs == null)
- {
- acceptEventArgs = new SocketAsyncEventArgs();
- acceptEventArgs.Completed += new EventHandler<SocketAsyncEventArgs>(AcceptEventArg_Completed);
- }
- else
- {
- acceptEventArgs.AcceptSocket = null; //释放上次绑定的Socket,等待下一个Socket连接
- }
- m_maxNumberAcceptedClients.WaitOne(); //获取信号量
- bool willRaiseEvent = listenSocket.AcceptAsync(acceptEventArgs);
- if (!willRaiseEvent)
- {
- ProcessAccept(acceptEventArgs);
- }
- }
接受连接响应事件代码:
- void AcceptEventArg_Completed(object sender, SocketAsyncEventArgs acceptEventArgs)
- {
- try
- {
- ProcessAccept(acceptEventArgs);
- }
- catch (Exception E)
- {
- Program.Logger.ErrorFormat("Accept client {0} error, message: {1}", acceptEventArgs.AcceptSocket, E.Message);
- Program.Logger.Error(E.StackTrace);
- }
- }
- private void ProcessAccept(SocketAsyncEventArgs acceptEventArgs)
- {
- Program.Logger.InfoFormat("Client connection accepted. Local Address: {0}, Remote Address: {1}",
- acceptEventArgs.AcceptSocket.LocalEndPoint, acceptEventArgs.AcceptSocket.RemoteEndPoint);
- AsyncSocketUserToken userToken = m_asyncSocketUserTokenPool.Pop();
- m_asyncSocketUserTokenList.Add(userToken); //添加到正在连接列表
- userToken.ConnectSocket = acceptEventArgs.AcceptSocket;
- userToken.ConnectDateTime = DateTime.Now;
- try
- {
- bool willRaiseEvent = userToken.ConnectSocket.ReceiveAsync(userToken.ReceiveEventArgs); //投递接收请求
- if (!willRaiseEvent)
- {
- lock (userToken)
- {
- ProcessReceive(userToken.ReceiveEventArgs);
- }
- }
- }
- catch (Exception E)
- {
- Program.Logger.ErrorFormat("Accept client {0} error, message: {1}", userToken.ConnectSocket, E.Message);
- Program.Logger.Error(E.StackTrace);
- }
- StartAccept(acceptEventArgs); //把当前异步事件释放,等待下次连接
- }
接受连接后,从当前Socket缓冲池AsyncSocketUserTokenPool中获取一个用户对象AsyncSocketUserToken,AsyncSocketUserToken包含一个接收异步事件m_receiveEventArgs,一个发送异步事件m_sendEventArgs,接收数据缓冲区m_receiveBuffer,发送数据缓冲区m_sendBuffer,协议逻辑调用对象m_asyncSocketInvokeElement,建立服务对象后,需要实现接收和发送的事件响应函数:
- void IO_Completed(object sender, SocketAsyncEventArgs asyncEventArgs)
- {
- AsyncSocketUserToken userToken = asyncEventArgs.UserToken as AsyncSocketUserToken;
- userToken.ActiveDateTime = DateTime.Now;
- try
- {
- lock (userToken)
- {
- if (asyncEventArgs.LastOperation == SocketAsyncOperation.Receive)
- ProcessReceive(asyncEventArgs);
- else if (asyncEventArgs.LastOperation == SocketAsyncOperation.Send)
- ProcessSend(asyncEventArgs);
- else
- throw new ArgumentException("The last operation completed on the socket was not a receive or send");
- }
- }
- catch (Exception E)
- {
- Program.Logger.ErrorFormat("IO_Completed {0} error, message: {1}", userToken.ConnectSocket, E.Message);
- Program.Logger.Error(E.StackTrace);
- }
- }
在Completed事件中需要处理发送和接收的具体逻辑代码,其中接收的逻辑实现如下:
- private void ProcessReceive(SocketAsyncEventArgs receiveEventArgs)
- {
- AsyncSocketUserToken userToken = receiveEventArgs.UserToken as AsyncSocketUserToken;
- if (userToken.ConnectSocket == null)
- return;
- userToken.ActiveDateTime = DateTime.Now;
- if (userToken.ReceiveEventArgs.BytesTransferred > 0 && userToken.ReceiveEventArgs.SocketError == SocketError.Success)
- {
- int offset = userToken.ReceiveEventArgs.Offset;
- int count = userToken.ReceiveEventArgs.BytesTransferred;
- if ((userToken.AsyncSocketInvokeElement == null) & (userToken.ConnectSocket != null)) //存在Socket对象,并且没有绑定协议对象,则进行协议对象绑定
- {
- BuildingSocketInvokeElement(userToken);
- offset = offset + 1;
- count = count - 1;
- }
- if (userToken.AsyncSocketInvokeElement == null) //如果没有解析对象,提示非法连接并关闭连接
- {
- Program.Logger.WarnFormat("Illegal client connection. Local Address: {0}, Remote Address: {1}", userToken.ConnectSocket.LocalEndPoint,
- userToken.ConnectSocket.RemoteEndPoint);
- CloseClientSocket(userToken);
- }
- else
- {
- if (count > 0) //处理接收数据
- {
- if (!userToken.AsyncSocketInvokeElement.ProcessReceive(userToken.ReceiveEventArgs.Buffer, offset, count))
- { //如果处理数据返回失败,则断开连接
- CloseClientSocket(userToken);
- }
- else //否则投递下次介绍数据请求
- {
- bool willRaiseEvent = userToken.ConnectSocket.ReceiveAsync(userToken.ReceiveEventArgs); //投递接收请求
- if (!willRaiseEvent)
- ProcessReceive(userToken.ReceiveEventArgs);
- }
- }
- else
- {
- bool willRaiseEvent = userToken.ConnectSocket.ReceiveAsync(userToken.ReceiveEventArgs); //投递接收请求
- if (!willRaiseEvent)
- ProcessReceive(userToken.ReceiveEventArgs);
- }
- }
- }
- else
- {
- CloseClientSocket(userToken);
- }
- }
由于我们制定的协议第一个字节是协议标识,因此在接收到第一个字节的时候需要绑定协议解析对象,具体代码实现如下:
- private void BuildingSocketInvokeElement(AsyncSocketUserToken userToken)
- {
- byte flag = userToken.ReceiveEventArgs.Buffer[userToken.ReceiveEventArgs.Offset];
- if (flag == (byte)SocketFlag.Upload)
- userToken.AsyncSocketInvokeElement = new UploadSocketProtocol(this, userToken);
- else if (flag == (byte)SocketFlag.Download)
- userToken.AsyncSocketInvokeElement = new DownloadSocketProtocol(this, userToken);
- else if (flag == (byte)SocketFlag.RemoteStream)
- userToken.AsyncSocketInvokeElement = new RemoteStreamSocketProtocol(this, userToken);
- else if (flag == (byte)SocketFlag.Throughput)
- userToken.AsyncSocketInvokeElement = new ThroughputSocketProtocol(this, userToken);
- else if (flag == (byte)SocketFlag.Control)
- userToken.AsyncSocketInvokeElement = new ControlSocketProtocol(this, userToken);
- else if (flag == (byte)SocketFlag.LogOutput)
- userToken.AsyncSocketInvokeElement = new LogOutputSocketProtocol(this, userToken);
- if (userToken.AsyncSocketInvokeElement != null)
- {
- Program.Logger.InfoFormat("Building socket invoke element {0}.Local Address: {1}, Remote Address: {2}",
- userToken.AsyncSocketInvokeElement, userToken.ConnectSocket.LocalEndPoint, userToken.ConnectSocket.RemoteEndPoint);
- }
- }
发送响应函数实现需要注意,我们是把发送数据放到一个列表中,当上一个发送事件完成响应Completed事件,这时我们需要检测发送队列中是否存在未发送的数据,如果存在则继续发送。
- private bool ProcessSend(SocketAsyncEventArgs sendEventArgs)
- {
- AsyncSocketUserToken userToken = sendEventArgs.UserToken as AsyncSocketUserToken;
- if (userToken.AsyncSocketInvokeElement == null)
- return false;
- userToken.ActiveDateTime = DateTime.Now;
- if (sendEventArgs.SocketError == SocketError.Success)
- return userToken.AsyncSocketInvokeElement.SendCompleted(); //调用子类回调函数
- else
- {
- CloseClientSocket(userToken);
- return false;
- }
- }
SendCompleted用于回调下次需要发送的数据,具体实现过程如下:
- public virtual bool SendCompleted()
- {
- m_activeDT = DateTime.UtcNow;
- m_sendAsync = false;
- AsyncSendBufferManager asyncSendBufferManager = m_asyncSocketUserToken.SendBuffer;
- asyncSendBufferManager.ClearFirstPacket(); //清除已发送的包
- int offset = 0;
- int count = 0;
- if (asyncSendBufferManager.GetFirstPacket(ref offset, ref count))
- {
- m_sendAsync = true;
- return m_asyncSocketServer.SendAsyncEvent(m_asyncSocketUserToken.ConnectSocket, m_asyncSocketUserToken.SendEventArgs,
- asyncSendBufferManager.DynamicBufferManager.Buffer, offset, count);
- }
- else
- return SendCallback();
- }
- //发送回调函数,用于连续下发数据
- public virtual bool SendCallback()
- {
- return true;
- }
当一个SocketAsyncEventArgs断开后,我们需要断开对应的Socket连接,并释放对应资源,具体实现函数如下:
- public void CloseClientSocket(AsyncSocketUserToken userToken)
- {
- if (userToken.ConnectSocket == null)
- return;
- string socketInfo = string.Format("Local Address: {0} Remote Address: {1}", userToken.ConnectSocket.LocalEndPoint,
- userToken.ConnectSocket.RemoteEndPoint);
- Program.Logger.InfoFormat("Client connection disconnected. {0}", socketInfo);
- try
- {
- userToken.ConnectSocket.Shutdown(SocketShutdown.Both);
- }
- catch (Exception E)
- {
- Program.Logger.ErrorFormat("CloseClientSocket Disconnect client {0} error, message: {1}", socketInfo, E.Message);
- }
- userToken.ConnectSocket.Close();
- userToken.ConnectSocket = null; //释放引用,并清理缓存,包括释放协议对象等资源
- m_maxNumberAcceptedClients.Release();
- m_asyncSocketUserTokenPool.Push(userToken);
- m_asyncSocketUserTokenList.Remove(userToken);
- }
3、SocketAsyncEventArgs封装和MSDN的不同点
MSDN在http://msdn.microsoft.com/zh-cn/library/system.NET.sockets.socketasynceventargs(v=vs.110).aspx实现了示例代码,并实现了初步的池化处理,我们是在它的基础上扩展实现了接收数据缓冲,发送数据队列,并把发送SocketAsyncEventArgs和接收SocketAsyncEventArgs分开,并实现了协议解析单元,这样做的好处是方便后续逻辑实现文件的上传,下载和日志输出。
DEMO下载地址:http://download.csdn.Net/detail/sqldebug_fan/7467745
免责声明:此代码只是为了演示C#完成端口编程,仅用于学习和研究,切勿用于商业用途。水平有限,C#也属于初学,错误在所难免,欢迎指正和指导。邮箱地址:fansheng_hx@163.com。
【转】C#高性能大容量SOCKET并发(二):SocketAsyncEventArgs封装的更多相关文章
- C#高性能大容量SOCKET并发(二):SocketAsyncEventArgs封装
原文:C#高性能大容量SOCKET并发(二):SocketAsyncEventArgs封装 1.SocketAsyncEventArgs介绍 SocketAsyncEventArgs是微软提供的高性能 ...
- C#高性能大容量SOCKET并发(十):SocketAsyncEventArgs线程模型
原文:C#高性能大容量SOCKET并发(十):SocketAsyncEventArgs线程模型 线程模型 SocketAsyncEventArgs编程模式不支持设置同时工作线程个数,使用的NET的IO ...
- C#高性能大容量SOCKET并发(转)
C#高性能大容量SOCKET并发(零):代码结构说明 C#高性能大容量SOCKET并发(一):IOCP完成端口例子介绍 C#高性能大容量SOCKET并发(二):SocketAsyncEventArgs ...
- C#高性能大容量SOCKET并发(零):代码结构说明
原文:C#高性能大容量SOCKET并发(零):代码结构说明 C#版完成端口具有以下特点: 连接在线管理(提供在线连接维护,连接会话管理,数据接收,连接断开等相关事件跟踪): 发送数据智能合并(组件会根 ...
- C#高性能大容量SOCKET并发(三):接收、发送
原文:C#高性能大容量SOCKET并发(三):接收.发送 异步数据接收有可能收到的数据不是一个完整包,或者接收到的数据超过一个包的大小,因此我们需要把接收的数据进行缓存.异步发送我们也需要把每个发送的 ...
- C#高性能大容量SOCKET并发(四):缓存设计
原文:C#高性能大容量SOCKET并发(四):缓存设计 在编写服务端大并发的应用程序,需要非常注意缓存设计,缓存的设计是一个折衷的结果,需要通过并发测试反复验证.有很多服务程序是在启动时申请足够的内存 ...
- C#高性能大容量SOCKET并发(十一):编写上传客户端
原文:C#高性能大容量SOCKET并发(十一):编写上传客户端 客户端封装整体框架 客户端编程基于阻塞同步模式,只有数据正常发送或接收才返回,如果发生错误则抛出异常,基于TcpClient进行封装,主 ...
- C#高性能大容量SOCKET并发(九):断点续传
原文:C#高性能大容量SOCKET并发(九):断点续传 上传断点续传 断点续传主要是用在上传或下载文件,一般做法是开始上传的时候,服务器返回上次已经上传的大小,如果上传完成,则返回-1:下载开始的时候 ...
- C#高性能大容量SOCKET并发(七):协议字符集
原文:C#高性能大容量SOCKET并发(七):协议字符集 UTF-8 UTF-8是UNICODE的一种变长字符编码又称万国码,由Ken Thompson于1992年创建.现在已经标准化为RFC 362 ...
- C#高性能大容量SOCKET并发(六):超时Socket断开(守护线程)和心跳包
原文:C#高性能大容量SOCKET并发(六):超时Socket断开(守护线程)和心跳包 守护线程 在服务端版Socket编程需要处理长时间没有发送数据的Socket,需要在超时多长时间后断开连接,我们 ...
随机推荐
- UVA 1328 - Period KMP
题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=36131 题意:给出一个长度为n的字符串,要求找到一些i,满足说从1 ...
- 小甲鱼PE详解之区块描述、对齐值以及RVA详解(PE详解06)
各种区块的描述: 很多朋友喜欢听小甲鱼的PE详解,因为他们觉得课堂上老师讲解的都是略略带过,绕得大家云里雾里~刚好小甲鱼文采也没课堂上的教授讲的那么好,只能以比较通俗的话语来给大家描述~ 通常,区块中 ...
- Android 电池电量进度条,上下滚动图片的进度条(battery)
最近,制作一个app,需要模拟一个电池电量的进度条,根据电量多少来设置百分比,进度条不断上下滚动,就像平时手机充电一样的电池电量进度条.我就自定义view实现了电量进度条.修改图片就可以达到自己想要的 ...
- 使用sqljdbc连接mssql数据库,maven生成jar运行后报"Exception in thread "main" java.lang.SecurityException"错误
错误信息如下: Exception in thread "main" java.lang.SecurityException: Invalid signature file dig ...
- mvc-4控制器和状态(1)
导语 将状态保存在客户端可以加快页面反映:但应当避免状态或数据保存在DOM中:在MVC中,状态应该保存在控制器中 控制器是视图和模型的纽带,只有控制器知道视图和模型的存在并将它们连接在一起:当加载页面 ...
- BZOJ2310 : ParkII
单路径最大和问题,设f[i][j][S]表示到达(i,j),轮廓线状态为S的最优解. S用4进制m+1位数表示,0表示无插头,1表示左括号,2表示右括号,3表示独立插头. 在DP之前先进行一次预处理, ...
- 【TYVJ】1463 - 智商问题(二分/分块)
http://tyvj.cn/Problem_Show.aspx?id=1463 二分的话是水题啊.. 为了学分块还是来写这题吧.. 二分: #include <cstdio> #incl ...
- SSH全注解开发
web.xml: <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi=&quo ...
- iOS开发项目之四 [ 调整自定义tabbar的位置与加号按钮的位置]
自定义tabbar与按钮的添加 01 - 把系统的tabbar用我们自己的覆盖 LHQTabBar *lhqTabBar = [[LHQTabBar alloc]init]; [self setVal ...
- QDialog, QFileDialog 和 QDesktopServices 的使用方法
Qt中的QDialog类是用来生成对话框的类,QFileDialog 类是QDialog的衍生类,主要用来生成打开文件,或是打开文件目录的对话框,或者是保存文件的对话框,下面我们一一来看代码: 1. ...