介绍
我最近需要为一个.net项目准备一个内部线程通信机制. 项目有多个使用ASP.NET,Windows 表单和控制台应用程序的服务器和客户端构成. 考虑到实现的可能性,我下定决心要使用原生的socket,而不是许多.NET中已经提前为我们构建好的组件, 像是所谓的管道, NetTcpClient 还有 Azure 服务总线.
这篇文章中的服务器基于System.Net.Sockets类异步方法. 这些允许你支持大量的socket客户端, 而一个客户端的连接是唯一的阻塞机制. 阻塞的时间是可以忽略不记得,所以服务器基本上是在当做一个多线程socket服务器在运作的.
背景
原生的socket在为你提供通信层面的完全控制权上具有优势, 而在处理不同的数据类型是具有很大的灵活性. 你甚至可以通过socket发送序列化了的CLR对象,尽管我在这里不会那样做. 这个项目将会想你展示如何在socket之间发送文本.
代码的运用
使用下面的代码,你初始化了一个Server类,并运行了Start()方法:
1 | 
Server myServer = new Server(); | 
 
 
 
如果你计划在一个Windows表单中管理服务器的话,我建议使用一个BackgroundWorker, 因为socket方法(一般会是ManualResentEvent) 将会阻塞GUI线程的运行.
Server 类:
01 | 
using System.Net.Sockets; | 
 
05 | 
    private static Socket listener; | 
 
06 | 
    public static ManualResetEvent allDone = new ManualResetEvent(false); | 
 
07 | 
    public const int _bufferSize = 1024; | 
 
08 | 
    public const int _port = 50000; | 
 
09 | 
    public static bool _isRunning = true; | 
 
13 | 
        public Socket workSocket = null; | 
 
14 | 
        public byte[] buffer = new byte[bufferSize]; | 
 
15 | 
        public StringBuilder sb = new StringBuilder(); | 
 
18 | 
    // Returns the string between str1 and str2 | 
 
19 | 
    static string Between(string str, string str1, string str2) | 
 
24 | 
        i1 = str.IndexOf(str1, StringComparison.InvariantCultureIgnoreCase); | 
 
27 | 
            i2 = str.IndexOf(str2, i1 + 1, StringComparison.InvariantCultureIgnoreCase); | 
 
30 | 
                rtn = str.Substring(i1 + str1.Length, i2 - i1 - str1.Length); | 
 
36 | 
    // Checks if the socket is connected | 
 
37 | 
    static bool IsSocketConnected(Socket s) | 
 
39 | 
        return !((s.Poll(1000, SelectMode.SelectRead) && (s.Available == 0)) || !s.Connected); | 
 
42 | 
    // Insert all the other methods here. | 
 
 
 
ManualResetEvent 是一个实现了你的socket服务器中事件的.NET类. 我们需要这个项目在我们想要发布阻塞操作的时候向代码发送信号. 你可以试验一下用bufferSize来适配你的需求. 如果能预期到消息的大小, 使用byte单位来设置消息的大小参数bufferSize. port是侦听TCP的端口参数. 要意识到为其它应用程序伺服所使用的接口. 如果你想要能够方便地停止服务器,你需要实现一些机制来将_isRunning设置成false. 这一般可以借助于使用一个 BackgroundWorker做到, 其中你可以使用myWorker.CancellationPending替换_isRunning. 我提到_isRunning的原因是给你在处理取消操作的问题上提供一个方向, 并向你展示侦听器可以方便的停止的.
Between() 和IsSocketConnected() 是辅助方法.
现在转过来看看方法. 首先是Start()方法:
03 | 
    IPHostEntry ipHostInfo = Dns.GetHostEntry(Dns.GetHostName()); | 
 
04 | 
    IPEndPoint localEP = new IPEndPoint(IPAddress.Any, _port); | 
 
05 | 
    listener = new Socket(localEP.Address.AddressFamily, SocketType.Stream, ProtocolType.Tcp); | 
 
06 | 
    listener.Bind(localEP); | 
 
12 | 
        listener.BeginAccept(new AsyncCallback(acceptCallback), listener); | 
 
13 | 
        bool isRequest = allDone.WaitOne(new TimeSpan(12, 0, 0));  // Blocks for 12 hours | 
 
18 | 
            // Do some work here every 12 hours | 
 
 
 
这个方法初始化了侦听器socket, 并开始等待用户连接的到来. 项目中主要的模式是使用异步委派. 异步委派是在调用者中的状态改变时被异步调用的方法. isRequest 告诉你WaitOne 是否已经因为有客户端连接或者超时而退出.
如果你有大量的客户端连接同时发生, 考虑提高Listen()方法的队列参数.
现在来看看下一个方法, acceptCallback . 这个方法由listener.BeginAccept异步调用. 当方法完成执行时,侦听器会立即侦听新的客户端.
01 | 
static void acceptCallback(IAsyncResult ar) | 
 
03 | 
    // Get the listener that handles the client request. | 
 
04 | 
    Socket listener = (Socket)ar.AsyncState; | 
 
08 | 
        Socket handler = listener.EndAccept(ar); | 
 
10 | 
        // Signal main thread to continue | 
 
14 | 
        StateObject state = new StateObject(); | 
 
15 | 
        state.workSocket = handler; | 
 
16 | 
        handler.BeginReceive(state.buffer, 0, _bufferSize, 0, new AsyncCallback(readCallback), state); | 
 
 
 
acceptCallback 会派生出另外一个异步指派: readCallback. 这个方法会读取来自socket的实际数据. 我已经为收发数据作了我自己的控制, 对于_bufferSize来说是不变的. 所有发送到服务器的字符串都必须用<!--SOCKET--> 和 <!--ENDSOCKET-->包起来. 同样,客户端在收到服务器的响应式,必须解除响应信息的包裹, 后者被<!--RESPONSE--> 和 <!--ENDRESPONSE-->包了起来。
01 | 
static void readCallback(IAsyncResult ar) | 
 
03 | 
    StateObject state = (StateObject)ar.AsyncState; | 
 
04 | 
    Socket handler = state.workSocket; | 
 
06 | 
    if (!IsSocketConnected(handler))  | 
 
12 | 
    int read = handler.EndReceive(ar); | 
 
14 | 
    // Data was read from the client socket. | 
 
17 | 
        state.sb.Append(Encoding.UTF8.GetString(state.buffer, 0, read)); | 
 
19 | 
        if (state.sb.ToString().Contains("<!--ENDSOCKET-->")) | 
 
22 | 
            string cmd = ts.Strings.Between(state.sb.ToString(), "<!--SOCKET-->", "<!--ENDSOCKET-->"); | 
 
27 | 
                    toSend = "How are you?"; | 
 
30 | 
                    toSend = "No I am not."; | 
 
34 | 
            toSend = "<!--RESPONSE-->" + toSend + "<!--ENDRESPONSE-->"; | 
 
36 | 
            byte[] bytesToSend = Encoding.UTF8.GetBytes(toSend); | 
 
37 | 
            handler.BeginSend(bytesToSend, 0, bytesToSend.Length, SocketFlags.None | 
 
38 | 
                , new AsyncCallback(sendCallback), state); | 
 
42 | 
            handler.BeginReceive(state.buffer, 0, _bufferSize, 0 | 
 
43 | 
                    , new AsyncCallback(readCallback), state); | 
 
 
 
readCallback 会派生另外一个方法, sendCallback, 它将会向客户端发送请求. 如果客户端没有关闭连接, sendCallback 将会向socket发送信号以获得更多的数据.
01 | 
static void sendCallback(IAsyncResult ar) | 
 
03 | 
    StateObject state = (StateObject)ar.AsyncState; | 
 
04 | 
    Socket handler = state.workSocket; | 
 
07 | 
    StateObject newstate = new StateObject(); | 
 
08 | 
    newstate.workSocket = handler; | 
 
09 | 
    handler.BeginReceive(newstate.buffer, 0, StateObject.BufferSize, 0, new AsyncCallback(readCallback), newstate); | 
 
 
 
我会将写一个socket客户端作为联系留给读者. socket客户端应该使用同异步调用同样的编程模式. 我希望你能从这篇文章中收获乐趣,并且会像一个socket程序员那样付诸实践!
要点
我在生产环境下使用了此代码,其中的socket服务器是一个自由文本搜索引擎。 SQL Server缺乏对自由文本搜索支持(你可以使用自由文本索引,但它们是缓慢和昂贵的)。socket服务器负载了大量导向IEnumerables的文本数据,并使用Linq来搜索文本。来自socket服务器的响应从数百万行的Unicode文本数据中搜索时间在几毫秒内。我们还使用了三个分布式的Sphinx服务器(www.sphinxsearch.com)。socket服务器充当了Sphinx服务器的高速缓存。如果你需要一个快速的自由文本搜索引擎,我强烈建议使用Sphinx。
												
												
						- 异步Socket服务器与客户端
		
  本文灵感来自Andre Azevedo 在CodeProject上面的一片文章,An Asynchronous Socket Server and Client,讲的是异步的Socket通信. S ...
		 
						- 可扩展多线程异步Socket服务器框架EMTASS 2.0   (转自:http://blog.csdn.net/hulihui)
		
可扩展多线程异步Socket服务器框架EMTASS 2.0 (转自:http://blog.csdn.net/hulihui) 0 前言 >>[前言].[第1节].[第2节].[第3节]. ...
		 
						- 可扩展多线程异步Socket服务器框架EMTASS 2.0 续
		
转载自Csdn:http://blog.csdn.net/hulihui/article/details/3158613 (原创文章,转载请注明来源:http://blog.csdn.net/huli ...
		 
						- C#实现的异步Socket服务器
		
介绍 我最近需要为一个.net项目准备一个内部线程通信机制. 项目有多个使用ASP.NET,Windows 表单和控制台应用程序的服务器和客户端构成. 考虑到实现的可能性,我下定决心要使用原生的soc ...
		 
								- 可扩展多线程异步Socket服务器框架EMTASS 2.0
		
0 前言 >>[前言].[第1节].[第2节].[第3节].[第4节].[第5节].[第6节] 在程序设计与实际应用中,Socket数据包接收服务器够得上一个经典问题了:需要计算机与网络编 ...
		 
						- 《Unity 3D游戏客户端基础框架》多线程异步 Socket 框架构建
		
引言: 之前写过一个 demo 案例大致讲解了 Socket 通信的过程,并和自建的服务器完成连接和简单的数据通信,详细的内容可以查看 Unity3D -- Socket通信(C#).但是在实际项目应 ...
		 
						- GJM :异步Socket [转载]
		
原帖地址:http://blog.csdn.net/awinye/article/details/537264 原文作者:Awinye 目录(?)[-] 转载请原作者联系 Overview of So ...
		 
						- Python简易聊天工具-基于异步Socket通信
		
继续学习Python中,最近看书<Python基础教程>中的虚拟茶话会项目,觉得很有意思,自己敲了一遍,受益匪浅,同时记录一下. 主要用到异步socket服务客户端和服务器模块asynco ...
		 
						- workerman是一个高性能的PHP socket服务器框架
		
workerman-chatorkerman是一款纯PHP开发的开源高性能的PHP socket服务器框架.被广泛的用于手机app.手游服务端.网络游戏服务器.聊天室服务器.硬件通讯服务器.智能家居. ...
		 
		
	
随机推荐
	
									- shell rename directory
			
mv can do two jobs. It can move files or directories It can rename files or directories To just rena ...
			 
						- [LOJ2542][PKUWC2018]随机游走(MinMax容斥+树形DP)
			
MinMax容斥将问题转化为求x到S中任意点的最小时间. 树形DP,直接求概率比较困难,考虑只求系数.最后由于x节点作为树根无父亲,所以求出的第二个系数就是答案. https://blog.csdn. ...
			 
						- 【洛谷】2602: [ZJOI2010]数字计数【数位DP】
			
P2602 [ZJOI2010]数字计数 题目描述 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 输入输出格式 输入格式: 输入文件中仅包含一行两个整数a ...
			 
						- Treap树理解
			
title: Treap树理解 comments: true date: 2016-10-06 07:57:37 categories: 算法 tags: Treap树 树 Treap树理解 简介 随 ...
			 
						- javascript中的lambda表达式
			
<!DOCTYPE html> <html> <head> </head> <body> <script> var nubLis ...
			 
						- 【原】Order属性决定了不同切面类中通知执行的先后顺序
			
[障碍再现] MyBatis配置多数据源时,数据源切换失败. [原因分析]    自定义切面和Spring自带事务切面“即<aop:advisor>”执行的先后顺序导致数据源不能切换成功. ...
			 
						- UIwebview   文件的下载与保存,以及mp3文件的播放
			
这里只是说说异步 单线程下载与文件的保存 以下载一个mp3文件并保存为例: -(void)loading { //设置文件下载地址 NSString *urlString = [NSString st ...
			 
						- 在当前的webview中跳转到新的url 使用WebView组件显示网页
			
如果希望点击链接由自己处理,而不是新开Android的系统browser中响应该链接.给WebView加一个事件监听对象(WebViewClient)并重写其中的一些方法:shouldOverride ...
			 
						- cocos2d-x项目101次相遇:使用触摸事件移动 精灵
			
cocos2d-x 101次相遇 / 文件夹  1   安装和环境搭建 -xcode  2   Scenes , Director, Layers, Sprites 3   建立图片菜单  4   在 ...
			 
						- 在MyEclipse上安装svn插件
			
最近需要用到myeclipse做一个商城的项目开发,用svn作为项目的版本控制软件.但是在myeclipse上安装svn插件就是装不好,反复折腾了好几次都安装不成功.网上提供的安装办法有两种,一是:在 ...