国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

使用C#來編寫一個異步的Socket服務器

2019-10-29 21:41:25
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了使用C#來編寫一個異步的Socket服務器,通過無阻塞機制來獲取更高的處理效率,需要的朋友可以參考下

介紹

我最近需要為一個.net項目準備一個內部線程通信機制. 項目有多個使用ASP.NET,Windows 表單和控制臺應用程序的服務器和客戶端構成. 考慮到實現的可能性,我下定決心要使用原生的socket,而不是許多.NET中已經提前為我們構建好的組件, 像是所謂的管道, NetTcpClient 還有 Azure 服務總線.

這篇文章中的服務器基于System.Net.Sockets類異步方法. 這些允許你支持大量的socket客戶端, 而一個客戶端的連接是唯一的阻塞機制. 阻塞的時間是可以忽略不記得,所以服務器基本上是在當做一個多線程socket服務器在運作的.

背景

原生的socket在為你提供通信層面的完全控制權上具有優勢, 而在處理不同的數據類型是具有很大的靈活性. 你甚至可以通過socket發送序列化了的CLR對象,盡管我在這里不會那樣做. 這個項目將會想你展示如何在socket之間發送文本.

代碼的運用

使用下面的代碼,你初始化了一個Server類,并運行了Start()方法:

 

 
  1. Server myServer = new Server(); 
  2. myServer.Start(); 

如果你計劃在一個Windows表單中管理服務器的話,我建議使用一個BackgroundWorker, 因為socket方法(一般會是ManualResentEvent) 將會阻塞GUI線程的運行.

Server 類:

 

 
  1. using System.Net.Sockets; 
  2.  
  3. public class Server 
  4. private static Socket listener; 
  5. public static ManualResetEvent allDone = new ManualResetEvent(false); 
  6. public const int _bufferSize = 1024; 
  7. public const int _port = 50000; 
  8. public static bool _isRunning = true
  9.  
  10. class StateObject 
  11. public Socket workSocket = null
  12. public byte[] buffer = new byte[bufferSize]; 
  13. public StringBuilder sb = new StringBuilder(); 
  14.  
  15. // Returns the string between str1 and str2 
  16. static string Between(string str, string str1, string str2) 
  17. int i1 = 0, i2 = 0; 
  18. string rtn = ""
  19.  
  20. i1 = str.IndexOf(str1, StringComparison.InvariantCultureIgnoreCase); 
  21. if (i1 > -1) 
  22. i2 = str.IndexOf(str2, i1 + 1, StringComparison.InvariantCultureIgnoreCase); 
  23. if (i2 > -1) 
  24. rtn = str.Substring(i1 + str1.Length, i2 - i1 - str1.Length); 
  25. return rtn; 
  26.  
  27. // Checks if the socket is connected 
  28. static bool IsSocketConnected(Socket s) 
  29. return !((s.Poll(1000, SelectMode.SelectRead) && (s.Available == 0)) || !s.Connected); 
  30.  
  31. // Insert all the other methods here. 

ManualResetEvent 是一個實現了你的socket服務器中事件的.NET類. 我們需要這個項目在我們想要發布阻塞操作的時候向代碼發送信號. 你可以試驗一下用bufferSize來適配你的需求. 如果能預期到消息的大小, 使用byte單位來設置消息的大小參數bufferSize. port是偵聽TCP的端口參數. 要意識到為其它應用程序伺服所使用的接口. 如果你想要能夠方便地停止服務器,你需要實現一些機制來將_isRunning設置成false. 這一般可以借助于使用一個 BackgroundWorker做到, 其中你可以使用myWorker.CancellationPending替換_isRunning. 我提到_isRunning的原因是給你在處理取消操作的問題上提供一個方向, 并向你展示偵聽器可以方便的停止的.

Between() 和IsSocketConnected() 是輔助方法.

現在轉過來看看方法. 首先是Start()方法:

 

 
  1. public void Start() 
  2. IPHostEntry ipHostInfo = Dns.GetHostEntry(Dns.GetHostName()); 
  3. IPEndPoint localEP = new IPEndPoint(IPAddress.Any, _port); 
  4. listener = new Socket(localEP.Address.AddressFamily, SocketType.Stream, ProtocolType.Tcp); 
  5. listener.Bind(localEP); 
  6.  
  7. while (_IsRunning) 
  8. allDone.Reset(); 
  9. listener.Listen(10); 
  10. listener.BeginAccept(new AsyncCallback(acceptCallback), listener); 
  11. bool isRequest = allDone.WaitOne(new TimeSpan(12, 0, 0)); // Blocks for 12 hours 
  12.  
  13. if (!isRequest) 
  14. allDone.Set(); 
  15. // Do some work here every 12 hours 
  16. listener.Close(); 

這個方法初始化了偵聽器socket, 并開始等待用戶連接的到來. 項目中主要的模式是使用異步委派. 異步委派是在調用者中的狀態改變時被異步調用的方法. isRequest 告訴你WaitOne 是否已經因為有客戶端連接或者超時而退出.

如果你有大量的客戶端連接同時發生, 考慮提高Listen()方法的隊列參數.

現在來看看下一個方法, acceptCallback . 這個方法由listener.BeginAccept異步調用. 當方法完成執行時,偵聽器會立即偵聽新的客戶端.

 

 
  1. static void acceptCallback(IAsyncResult ar) 
  2. // Get the listener that handles the client request. 
  3. Socket listener = (Socket)ar.AsyncState; 
  4.  
  5. if (listener != null
  6. Socket handler = listener.EndAccept(ar); 
  7.  
  8. // Signal main thread to continue 
  9. allDone.Set(); 
  10.  
  11. // Create state 
  12. StateObject state = new StateObject(); 
  13. state.workSocket = handler; 
  14. handler.BeginReceive(state.buffer, 0, _bufferSize, 0, new AsyncCallback(readCallback), state); 

acceptCallback 會派生出另外一個異步指派: readCallback. 這個方法會讀取來自socket的實際數據. 我已經為收發數據作了我自己的控制, 對于_bufferSize來說是不變的. 所有發送到服務器的字符串都必須用包起來. 同樣,客戶端在收到服務器的響應式,必須解除響應信息的包裹, 后者被包了起來。

 

 
  1. static void readCallback(IAsyncResult ar) 
  2. StateObject state = (StateObject)ar.AsyncState; 
  3. Socket handler = state.workSocket; 
  4.  
  5. if (!IsSocketConnected(handler))  
  6. handler.Close(); 
  7. return
  8.  
  9. int read = handler.EndReceive(ar); 
  10.  
  11. // Data was read from the client socket. 
  12. if (read > 0) 
  13. state.sb.Append(Encoding.UTF8.GetString(state.buffer, 0, read)); 
  14.  
  15. if (state.sb.ToString().Contains("<!--ENDSOCKET-->")) 
  16. string toSend = ""
  17. string cmd = ts.Strings.Between(state.sb.ToString(), "<!--SOCKET-->""<!--ENDSOCKET-->"); 
  18.  
  19. switch (cmd) 
  20. case "Hi!"
  21. toSend = "How are you?"
  22. break
  23. case "Milky Way?"
  24. toSend = "No I am not."
  25. break
  26.  
  27. toSend = "<!--RESPONSE-->" + toSend + "<!--ENDRESPONSE-->"
  28.  
  29. byte[] bytesToSend = Encoding.UTF8.GetBytes(toSend); 
  30. handler.BeginSend(bytesToSend, 0, bytesToSend.Length, SocketFlags.None 
  31. new AsyncCallback(sendCallback), state); 
  32. else 
  33. handler.BeginReceive(state.buffer, 0, _bufferSize, 0 
  34. new AsyncCallback(readCallback), state); 
  35. else 
  36. handler.Close(); 

readCallback 會派生另外一個方法, sendCallback, 它將會向客戶端發送請求. 如果客戶端沒有關閉連接, sendCallback 將會向socket發送信號以獲得更多的數據.

 

 
  1. static void sendCallback(IAsyncResult ar) 
  2. StateObject state = (StateObject)ar.AsyncState; 
  3. Socket handler = state.workSocket; 
  4. handler.EndSend(ar); 
  5.  
  6. StateObject newstate = new StateObject(); 
  7. newstate.workSocket = handler; 
  8. 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。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 佳木斯市| 静宁县| 阳原县| 望江县| 绥德县| 自治县| 辽阳县| 上栗县| 海口市| 磐石市| 易门县| 梨树县| 江都市| 岳阳市| 商丘市| 饶平县| 汉沽区| 东乡族自治县| 舞钢市| 满洲里市| 金塔县| 张家口市| 油尖旺区| 芒康县| 梨树县| 大悟县| 通榆县| 抚顺市| 垫江县| 吉隆县| 饶平县| 衡南县| 鹿泉市| 曲阳县| 峨山| 镇平县| 广汉市| 县级市| 神木县| 洱源县| 新乐市|