介绍
  我最近需要为一个。net项目准备一个内部线程通信机制。 项目有多个使用ASP.NET,Windows 表单和控制台应用程序的服务器和客户端构成。 考虑到实现的可能性,我下定决心要使用原生的socket,而不是许多。NET中已经提前为我们构建好的组件, 像是所谓的管道, NetTcpClient 还有 Azure 服务总线。
  这篇文章中的服务器基于System.Net.Sockets类异步方法。 这些允许你支持大量的socket客户端, 而一个客户端的连接是唯一的阻塞机制。 阻塞的时间是可以忽略不记得,所以服务器基本上是在当做一个多线程socket服务器在运作的。
  背景
  原生的socket在为你提供通信层面的完全控制权上具有优势, 而在处理不同的数据类型是具有很大的灵活性。 你甚至可以通过socket发送序列化了的CLR对象,尽管我在这里不会那样做。 这个项目将会想你展示如何在socket之间发送文本。
  代码的运用
  使用下面的代码,你初始化了一个Server类,并运行了Start()方法:
  1Server myServer = new Server();
  2myServer.Start();
  如果你计划在一个Windows表单中管理服务器的话,我建议使用一个BackgroundWorker, 因为socket方法(一般会是ManualResentEvent) 将会阻塞GUI线程的运行。
  Server 类:
  01using System.Net.Sockets;
  02
  03public class Server
  04{
  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;
  10
  11 class StateObject
  12 {
  13 public Socket workSocket = null;
  14 public byte[] buffer = new byte[bufferSize];
  15 public StringBuilder sb = new StringBuilder();
  16 }
  17
  18 // Returns the string between str1 and str2
  19 static string Between(string str, string str1, string str2)
  20 {
  21 int i1 = 0, i2 = 0;
  22 string rtn = "";
  23
  24 i1 = str.IndexOf(str1, StringComparison.InvariantCultureIgnoreCase);
  25 if (i1 > -1)
  26 {
  27 i2 = str.IndexOf(str2, i1 + 1, StringComparison.InvariantCultureIgnoreCase);
  28 if (i2 > -1)
  29 {
  30 rtn = str.Substring(i1 + str1.Length, i2 - i1 - str1.Length);
  31 }
  32 }
  33 return rtn;
  34 }
  35
  36 // Checks if the socket is connected
  37 static bool IsSocketConnected(Socket s)
  38 {
  39 return !((s.Poll(1000, SelectMode.SelectRead) && (s.Available == 0)) || !s.Connected);
  40 }
  41
  42 // Insert all the other methods here.
  43}
  ManualResetEvent 是一个实现了你的socket服务器中事件的。NET类。 我们需要这个项目在我们想要发布阻塞操作的时候向代码发送信号。 你可以试验一下用bufferSize来适配你的需求。 如果能预期到消息的大小, 使用byte单位来设置消息的大小参数bufferSize. port是侦听TCP的端口参数。 要意识到为其它应用程序伺服所使用的接口。 如果你想要能够方便地停止服务器,你需要实现一些机制来将_isRunning设置成false. 这一般可以借助于使用一个 BackgroundWorker做到, 其中你可以使用myWorker.CancellationPending替换_isRunning. 我提到_isRunning的原因是给你在处理取消操作的问题上提供一个方向, 并向你展示侦听器可以方便的停止的 www.jamo123.com
  Between() 和IsSocketConnected() 是辅助方法。
  现在转过来看看方法。 首先是Start()方法:
  01public void Start()
  02{
  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);
  07
  08 while (_IsRunning)
  09 {
  10 allDone.Reset();
  11 listener.Listen(10);
  12 listener.BeginAccept(new AsyncCallback(acceptCallback), listener);
  13 bool isRequest = allDone.WaitOne(new TimeSpan(12, 0, 0)); // Blocks for 12 hours
  14
  15 if (!isRequest)
  16 {
  17 allDone.Set();
  18 // Do some work here every 12 hours
  19 }
  20 }
  21 listener.Close();
  22}
  这个方法初始化了侦听器socket, 并开始等待用户连接的到来。 项目中主要的模式是使用异步委派。 异步委派是在调用者中的状态改变时被异步调用的方法。 isRequest 告诉你WaitOne 是否已经因为有客户端连接或者超时而退出。
  如果你有大量的客户端连接同时发生, 考虑提高Listen()方法的队列参数。
  现在来看看下一个方法, acceptCallback . 这个方法由listener.BeginAccept异步调用。 当方法完成执行时,侦听器会立即侦听新的客户端。
  01static void acceptCallback(IAsyncResult ar)
  02{
  03 // Get the listener that handles the client request.
  04 Socket listener = (Socket)ar.AsyncState;
  05
  06 if (listener != null)
  07 {
  08 Socket handler = listener.EndAccept(ar);
  09
  10 // Signal main thread to continue
  11 allDone.Set();
  12
  13 // Create state
  14 StateObject state = new StateObject();
  15 state.workSocket = handler;
  16 handler.BeginReceive(state.buffer, 0, _bufferSize, 0, new AsyncCallback(readCallback), state);
  17 }
  18}
  acceptCallback 会派生出另外一个异步指派: readCallback. 这个方法会读取来自socket的实际数据。 我已经为收发数据作了我自己的控制, 对于_bufferSize来说是不变的。 所有发送到服务器的字符串都必须用<!--SOCKET--> 和 <!--ENDSOCKET-->包起来。 同样,客户端在收到服务器的响应式,必须解除响应信息的包裹, 后者被<!--RESPONSE--> 和 <!--ENDRESPONSE-->包了起来 www.tygj123.com
  01static void readCallback(IAsyncResult ar)
  02{
  03 StateObject state = (StateObject)ar.AsyncState;
  04 Socket handler = state.workSocket;
  05
  06 if (!IsSocketConnected(handler))
  07 {
  08 handler.Close();
  09 return;
  10 }
  11
  12 int read = handler.EndReceive(ar);
  13
  14 // Data was read from the client socket.
  15 if (read > 0)
  16 {
  17 state.sb.Append(Encoding.UTF8.GetString(state.buffer, 0, read));
  18
  19 if (state.sb.ToString()。Contains("<!--ENDSOCKET-->"))
  20 {
  21 string toSend = "";
  22 string cmd = ts.Strings.Between(state.sb.ToString(), "<!--SOCKET-->", "<!--ENDSOCKET-->");
  23
  24 switch (cmd)
  25 {
  26 case "Hi!":
  27 toSend = "How are you?";
  28 break;
  29 case "Milky Way?":
  30 toSend = "No I am not.";
  31 break;
  32 }
  33
  34 toSend = "<!--RESPONSE-->" + toSend + "<!--ENDRESPONSE-->";
  35
  36 byte[] bytesToSend = Encoding.UTF8.GetBytes(toSend);
  37 handler.BeginSend(bytesToSend, 0, bytesToSend.Length, SocketFlags.None
  38 , new AsyncCallback(sendCallback), state);
  39 }
  40 else
  41 {
  42 handler.BeginReceive(state.buffer, 0, _bufferSize, 0
  43 , new AsyncCallback(readCallback), state);
  44 }
  45 }
  46 else
  47 {
  48 handler.Close();
  49 }
  50}
  readCallback 会派生另外一个方法, sendCallback, 它将会向客户端发送请求。 如果客户端没有关闭连接, sendCallback 将会向socket发送信号以获得更多的数据。
  01static void sendCallback(IAsyncResult ar)
  02{
  03 StateObject state = (StateObject)ar.AsyncState;
  04 Socket handler = state.workSocket;
  05 handler.EndSend(ar);
  06
  07 StateObject newstate = new StateObject();
  08 newstate.workSocket = handler;
  09 handler.BeginReceive(newstate.buffer, 0, StateObject.BufferSize, 0, new AsyncCallback(readCallback), newstate);
  10}
  我会将写一个socket客户端作为联系留给读者。 socket客户端应该使用同异步调用同样的编程模式。 我希望你能从这篇文章中收获乐趣,并且会像一个socket程序员那样付诸实践!
  要点
  我在生产环境下使用了此代码,其中的socket服务器是一个自由文本搜索引擎。 SQL Server缺乏对自由文本搜索支持(你可以使用自由文本索引,但它们是缓慢和昂贵的)。socket服务器负载了大量导向IEnumerables的文本数据,并使用Linq来搜索文本。来自socket服务器的响应从数百万行的Unicode文本数据中搜索时间在几毫秒内。我们还使用了三个分布式的Sphinx服务器。socket服务器充当了Sphinx服务器的高速缓存 www.yztrans.com

C#实现的异步Socket服务器的更多相关文章

  1. 异步Socket服务器与客户端

      本文灵感来自Andre Azevedo 在CodeProject上面的一片文章,An Asynchronous Socket Server and Client,讲的是异步的Socket通信. S ...

  2. 可扩展多线程异步Socket服务器框架EMTASS 2.0 (转自:http://blog.csdn.net/hulihui)

    可扩展多线程异步Socket服务器框架EMTASS 2.0 (转自:http://blog.csdn.net/hulihui) 0 前言 >>[前言].[第1节].[第2节].[第3节]. ...

  3. 可扩展多线程异步Socket服务器框架EMTASS 2.0 续

    转载自Csdn:http://blog.csdn.net/hulihui/article/details/3158613 (原创文章,转载请注明来源:http://blog.csdn.net/huli ...

  4. C# 实现的异步 Socket 服务器

    介绍 我最近需要为一个.net项目准备一个内部线程通信机制. 项目有多个使用ASP.NET,Windows 表单和控制台应用程序的服务器和客户端构成. 考虑到实现的可能性,我下定决心要使用原生的soc ...

  5. 可扩展多线程异步Socket服务器框架EMTASS 2.0

    0 前言 >>[前言].[第1节].[第2节].[第3节].[第4节].[第5节].[第6节] 在程序设计与实际应用中,Socket数据包接收服务器够得上一个经典问题了:需要计算机与网络编 ...

  6. 《Unity 3D游戏客户端基础框架》多线程异步 Socket 框架构建

    引言: 之前写过一个 demo 案例大致讲解了 Socket 通信的过程,并和自建的服务器完成连接和简单的数据通信,详细的内容可以查看 Unity3D -- Socket通信(C#).但是在实际项目应 ...

  7. GJM :异步Socket [转载]

    原帖地址:http://blog.csdn.net/awinye/article/details/537264 原文作者:Awinye 目录(?)[-] 转载请原作者联系 Overview of So ...

  8. Python简易聊天工具-基于异步Socket通信

    继续学习Python中,最近看书<Python基础教程>中的虚拟茶话会项目,觉得很有意思,自己敲了一遍,受益匪浅,同时记录一下. 主要用到异步socket服务客户端和服务器模块asynco ...

  9. workerman是一个高性能的PHP socket服务器框架

    workerman-chatorkerman是一款纯PHP开发的开源高性能的PHP socket服务器框架.被广泛的用于手机app.手游服务端.网络游戏服务器.聊天室服务器.硬件通讯服务器.智能家居. ...

随机推荐

  1. bzoj2400

    首先xor类的题目一定要逐位考虑,因为位位之间是不相互影响的逐位考虑每个点是0还是1,这就转化成了一个这样一个问题对于每个点可以选择属于S集合(这位是0)或T集合(这位是1)某些的点对(一条边的两端) ...

  2. hiho #1044 : 状态压缩·一

    描述 小Hi和小Ho在兑换到了喜欢的奖品之后,便继续起了他们的美国之行,思来想去,他们决定乘坐火车前往下一座城市——那座城市即将举行美食节! 但是不幸的是,小Hi和小Ho并没有能够买到很好的火车票—— ...

  3. HDU 4612 Warm up(手动扩栈,求树上哪两个点的距离最远)

    题目大意: 给你一个无向图,问加一条边之后最少还剩下几座桥. (注意重边处理)   分析:其实当我们把边双连通分量给求出来之后我们就能将连通块求出来,这样我们就可以重新构图.重新构造出来的图肯定是一颗 ...

  4. HDU 3829 Cat VS Dog(最大独立集)

    题目大意: 有n只猫,有m只狗.现在有P个学生去参观动物园.每个孩子有喜欢的动物和不喜欢的动物.假如他喜欢猫那么他就一定不喜欢狗(反之亦然). 如果一个孩子喜欢一个动物,那么这个动物不会被移除,若是不 ...

  5. 数据结构(线段树):NOI 2016 区间

    [问题描述] [输入格式] [输出格式] [样例输入] 6 3 3 5 1 2 3 4 2 2 1 5 1 4 [样例输出] 2 [样例说明] [更多样例] 下载 [样例 2 输入输出] 见目录下的 ...

  6. 字符串(后缀数组):POJ 3294 Life Forms

    Life Forms Description You may have wondered why most extraterrestrial life forms resemble humans, d ...

  7. server 2008 ftp 环境重点说明

    最近 在弄ftp  环境,但是 到server 2008 r2  这个系统之后,按照之前的方法 不行了 具体情况如下 利用本机 资源管理器 访问不了,根本不出现 登录框 提示 然后 到ftp  站点 ...

  8. 数据结构——HDU1312:Red and Black(DFS)

    题目描述 There is a rectangular room, covered with square tiles. Each tile is colored either red or blac ...

  9. Geometric Shapes (poj3449多边形相交)

    题意:给你一些多边形的点,判断每个多边形和那些多边形相交,编号按照字典序输出 思路:枚举每个多边形的每条边看是否相交,这里的相交是包括端点的,关键是给你正方形不相邻两个点求另外两个点怎么求,长方形给你 ...

  10. cat-mvc 一个nodejs mvc 框架

    不多说,自己看,哈哈哈 https://www.npmjs.com/package/cat-mvc