C# socket编程实践——简单聊天室

本文介绍了一种基于C#的TCP服务器的设计与实现方法,详细阐述了如何搭建服务器、处理客户端连接请求、广播消息等核心功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

整体结构

关于怎么具体一步步使用socket我就不说了,有兴趣同学可以看看你得学会并且学得会的Socket编程基础知识,看看我们服务器的结构,我写了一个TcpHelper类来处理服务器操作

首先定义 一个ClientInfo类存放Client信息

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. /// <summary>  
  2.         /// 启动服务器,监听客户端请求  
  3.         /// </summary>  
  4.         /// <param name="port">服务器端进程口号</param>  
  5.         public void Run(int port);  
  6.   
  7. /// <summary>  
  8.         /// 在独立线程中不停地向所有客户端广播消息  
  9.         /// </summary>  
  10.         private void Broadcast();  
  11.   
  12. /// <summary>  
  13.         /// 把客户端消息打包处理(拼接上谁什么时候发的什么消息)  
  14.         /// </summary>  
  15.         /// <returns>The message.</returns>  
  16.         /// <param name="sm">Sm.</param>  
  17.         private byte[] PackageMessage(SocketMessage sm);  
  18.   
  19. /// <summary>  
  20.         /// 处理客户端连接请求,成功后把客户端加入到clientPool  
  21.         /// </summary>  
  22.         /// <param name="result">Result.</param>  
  23.         private void Accept(IAsyncResult result);  
  24.   
  25. /// <summary>  
  26.         /// 处理客户端发送的消息,接收成功后加入到msgPool,等待广播  
  27.         /// </summary>  
  28.         /// <param name="result">Result.</param>  
  29.         private void Recieve(IAsyncResult result);  
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. 逐个分析一下把  
  2.   
  3. void run(int port)  
  4. 这是该类唯一提供的共有方法,供外界调用,来根据port参数创建一个socket  


[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. public class ClientInfo  
  2.     {  
  3.         public byte[] buffer;  
  4.         public string NickName { getset; }  
  5.         public EndPoint Id { getset; }  
  6.         public IntPtr handle { getset; }  
  7.         public string Name  
  8.         {  
  9.             get  
  10.             {  
  11.                 if (!string.IsNullOrEmpty(NickName))  
  12.                 {  
  13.                     return NickName;  
  14.                 }  
  15.                 else  
  16.                 {  
  17.                     return string.Format("{0}#{1}", Id, handle);  
  18.                 }  
  19.             }  
  20.         }  
  21.     }  

然后是一个SocketMessage类,记录客户端发来的消息

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. public class SocketMessage  
  2.     {  
  3.         public bool isLogin { getset; }  
  4.         public ClientInfo Client { getset; }  
  5.         public string Message { getset; }  
  6.         public DateTime Time { getset; }  
  7.     }  

然后定义两个全局变量记录所有客户端及所有客户端发来的消息

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. private Dictionary<Socket, ClientInfo> clientPool = new Dictionary<Socket, ClientInfo>();  
  2. private List<SocketMessage> msgPool = new List<SocketMessage>();  
然后就是几个主要方法的定义

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. /// <summary>  
  2.         /// 启动服务器,监听客户端请求  
  3.         /// </summary>  
  4.         /// <param name="port">服务器端进程口号</param>  
  5.         public void Run(int port);  
  6.   
  7. /// <summary>  
  8.         /// 在独立线程中不停地向所有客户端广播消息  
  9.         /// </summary>  
  10.         private void Broadcast();  
  11.   
  12. /// <summary>  
  13.         /// 把客户端消息打包处理(拼接上谁什么时候发的什么消息)  
  14.         /// </summary>  
  15.         /// <returns>The message.</returns>  
  16.         /// <param name="sm">Sm.</param>  
  17.         private byte[] PackageMessage(SocketMessage sm);  
  18.   
  19. /// <summary>  
  20.         /// 处理客户端连接请求,成功后把客户端加入到clientPool  
  21.         /// </summary>  
  22.         /// <param name="result">Result.</param>  
  23.         private void Accept(IAsyncResult result);  
  24.   
  25. /// <summary>  
  26.         /// 处理客户端发送的消息,接收成功后加入到msgPool,等待广播  
  27.         /// </summary>  
  28.         /// <param name="result">Result.</param>  
  29.         private void Recieve(IAsyncResult result);  

逐个分析一下把

void run(int port)

这是该类唯一提供的共有方法,供外界调用,来根据port参数创建一个socket

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. public void Run(int port)  
  2.         {  
  3.             Thread serverSocketThraed = new Thread(() =>  
  4.             {  
  5.                 Socket server = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);  
  6.                 server.Bind(new IPEndPoint(IPAddress.Any, port));  
  7.                 server.Listen(10);  
  8.                 server.BeginAccept(new AsyncCallback(Accept), server);  
  9.             });  
  10.   
  11.             serverSocketThraed.Start();  
  12.             Console.WriteLine("Server is ready");  
  13.             Broadcast();  
  14.         }  

 代码很简单,需要注意的有几点

1.在一个新线程中创建服务器socket,最多允许10个客户端连接。

2.在方法最后调用Broadcast()方法用于向所有客户端广播消息

3.BeginAccept方法,MSDN上有权威解释,但是觉得不够接地气,简单说一下我的理解,首先这个方法是异步的,用于服务器接受一个客户端的连接,第一个参数实际上是回调函数,在C#中使用委托,在回调函数中通过调用EndAccept就可以获得尝试连接的客户端socket,第二个参数是包含请求state的对象,传入server socket对象本身就可以了

void Accept(IAsyncResult result)

方法用于处理客户端连接请求

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. private void Accept(IAsyncResult result)  
  2.         {  
  3.             Socket server = result.AsyncState as Socket;  
  4.             Socket client = server.EndAccept(result);  
  5.             try  
  6.             {  
  7.                 //处理下一个客户端连接  
  8.                 server.BeginAccept(new AsyncCallback(Accept), server);  
  9.                 byte[] buffer = new byte[1024];  
  10.                 //接收客户端消息  
  11.                 client.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(Recieve), client);  
  12.                 ClientInfo info = new ClientInfo();  
  13.                 info.Id = client.RemoteEndPoint;  
  14.                 info.handle = client.Handle;  
  15.                 info.buffer = buffer;  
  16.                 //把客户端存入clientPool  
  17.                 this.clientPool.Add(client, info);  
  18.                 Console.WriteLine(string.Format("Client {0} connected", client.RemoteEndPoint));  
  19.             }  
  20.             catch (Exception ex)  
  21.             {  
  22.                 Console.WriteLine("Error :\r\n\t" + ex.ToString());  
  23.             }  
  24.         }  


 BeginRecieve方法的MSDN有解释,和Accept一样也是异步处理,接收客户端消息,放入第一个参数中,它也传入了一个回调函数的委托,和带有socket state的对象,用于处理下一次接收。我们把接收成功地客户端socket及其对应信息存放到clientPool中

void Recieve(IAsyncResult result)

方法用于接收客户端消息,并把所有消息及其发送者信息存入msgInfo,等待广播

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. private void Recieve(IAsyncResult result)  
  2.         {  
  3.             Socket client = result.AsyncState as Socket;  
  4.   
  5.             if (client == null || !clientPool.ContainsKey(client))  
  6.             {  
  7.                 return;  
  8.             }  
  9.   
  10.             try  
  11.             {  
  12.                 int length = client.EndReceive(result);  
  13.                 byte[] buffer = clientPool[client].buffer;  
  14.   
  15.                 //接收消息  
  16.                 client.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(Recieve), client);  
  17.                 string msg = Encoding.UTF8.GetString(buffer, 0, length);  
  18.                 SocketMessage sm = new SocketMessage();  
  19.                 sm.Client = clientPool[client];  
  20.                 sm.Time = DateTime.Now;  
  21.   
  22.                 Regex reg = new Regex(@"{<(.*?)>}");  
  23.                 Match m = reg.Match(msg);  
  24.                 if (m.Value != ""//处理客户端传来的用户名  
  25.                 {  
  26.                     clientPool[client].NickName = Regex.Replace(m.Value, @"{<(.*?)>}""$1");  
  27.                     sm.isLogin = true;  
  28.                     sm.Message = "login!";  
  29.                     Console.WriteLine("{0} login @ {1}", client.RemoteEndPoint,DateTime.Now);  
  30.                 }  
  31.                 else //处理客户端传来的普通消息  
  32.                 {  
  33.                     sm.isLogin = false;  
  34.                     sm.Message = msg;  
  35.                     Console.WriteLine("{0} @ {1}\r\n    {2}", client.RemoteEndPoint,DateTime.Now,msg);  
  36.                 }  
  37.                 msgPool.Add(sm);  
  38.             }  
  39.             catch  
  40.             {  
  41.                 //把客户端标记为关闭,并在clientPool中清除  
  42.                 client.Disconnect(true);  
  43.                 Console.WriteLine("Client {0} disconnet", clientPool[client].Name);  
  44.                 clientPool.Remove(client);  
  45.             }  
  46.         }  

这个的代码都很简单,就不多解释了,我加入了用户名处理用于广播客户端消息的时候显示客户端自定义的昵称而不是生硬的ip地址+端口号,当然这里需要客户端配合

  Broadcast()

服务器已经和客户端连接成功,并且接收到了客户端消息,我们就可以看看该怎么广播消息了,Broadcast()方法已经在run()方法内调用,看看它是怎么运作广播客户端消息的

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. private void Broadcast()  
  2.         {  
  3.             Thread broadcast = new Thread(() =>  
  4.             {  
  5.                 while (true)  
  6.                 {  
  7.                     if (msgPool.Count > 0)  
  8.                     {  
  9.                         byte[] msg = PackageMessage(msgPool[0]);  
  10.                         foreach (KeyValuePair<Socket, ClientInfo> cs in clientPool)  
  11.                         {  
  12.                             Socket client = cs.Key;  
  13.                             if (client.Connected)  
  14.                             {  
  15.                                 client.Send(msg, msg.Length, SocketFlags.None);  
  16.                             }  
  17.                         }  
  18.                         msgPool.RemoveAt(0);  
  19.                     }  
  20.                 }  
  21.             });  
  22.   
  23.             broadcast.Start();  
  24.         }  

Broadcast()方法启用了一个新线程,循环检测msgPool是否为空,当不为空的时候遍历所有客户端,调用send方法发送msgPool里面的第一条消息,然后清除该消息继续检测,直到消息广播完,其实这就是一个阉割版的观察者模式 ,顺便看一下打包数据方法
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. private byte[] PackageMessage(SocketMessage sm)  
  2.         {  
  3.             StringBuilder packagedMsg = new StringBuilder();  
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. <span style="white-space:pre">    </span>   if(sm!=null)  
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. <span style="white-space:pre">    </span>  {  
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. if (!sm.isLogin) //消息是login信息  
  2. {  
  3.     packagedMsg.AppendFormat("{0} @ {1}:\r\n    ", sm.Client.Name, sm.Time.ToShortTimeString());  
  4.     packagedMsg.Append(sm.Message);  
  5. }  
  6. else //处理普通消息  
  7. {  
  8.     packagedMsg.AppendFormat("{0} login @ {1}", sm.Client.Name, sm.Time.ToShortTimeString());  
  9. }  
  10. le="white-space:pre">    </span>  <pre name="code" class="csharp">          }  
return Encoding.UTF8.GetBytes(packagedMsg.ToString()); }
 

如何使用

[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. static void Main(string[] args)  
  2.         {  
  3.             TcpHelper helper = new TcpHelper();  
  4.             helper.Run(8080);  
  5.         }  

这样我们就启用了server,看看简单的客户端实现,原理类似,不再分析了
[csharp]  view plain  copy
 print ? 在CODE上查看代码片 派生到我的代码片
  1. class Program  
  2.     {  
  3.         private static byte[] buf = new byte[1024];  
  4.         static void Main(string[] args)  
  5.         {  
  6.             Console.Write("Enter your name: ");  
  7.             string name = Console.ReadLine();  
  8.             Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);  
  9.             client.Connect(new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8080));  
  10.             Console.WriteLine("Connected to server, enter $q to quit");  
  11.             name = "{<" + name.Trim() + ">}";  
  12.             byte[] nameBuf = Encoding.UTF8.GetBytes(name);  
  13.             client.BeginSend(nameBuf, 0, nameBuf.Length, SocketFlags.None, nullnull);  
  14.             client.BeginReceive(buf, 0, buf.Length, SocketFlags.None, new AsyncCallback(Recieve), client);  
  15.             while (true)  
  16.             {  
  17.                 string msg = Console.ReadLine();  
  18.                 if (msg == "$q")   
  19.                 {  
  20.                     client.Close();  
  21.                     break;  
  22.                 }  
  23.                 byte[] output = Encoding.UTF8.GetBytes(msg);  
  24.                 client.BeginSend(output, 0, output.Length, SocketFlags.None, nullnull);  
  25.             }  
  26.             Console.Write("Disconnected. Press any key to exit... ");  
  27.             Console.ReadKey();  
  28.         }  
  29.   
  30.         private static void Recieve(IAsyncResult result)  
  31.         {  
  32.             try  
  33.             {  
  34.                 Socket client = result.AsyncState as Socket;  
  35.                 int length = client.EndReceive(result);  
  36.                 string msg = Encoding.UTF8.GetString(buf, 0, length);  
  37.                 Console.WriteLine(msg);  
  38.                 client.BeginReceive(buf, 0, buf.Length, SocketFlags.None, new AsyncCallback(Recieve), client);  
  39.             }  
  40.             catch  
  41.             {  
  42.             }  
  43.         }  
  44.     }  

源码下载地址:http://download.youkuaiyun.com/detail/flyaurora/9396878

内容转载自:http://www.cnblogs.com/dolphinX/p/3462496.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值