当前位置: 首页 > news >正文

C# Socket 通信:异步、同步与断线重连

1、C#Socket异步、同步通信服务端、客户端源码; 2、断线重连(服务端或客户端没有启动顺序要求,先开启的等待另一端连接);3、服务端支持同时连接多个客户端;4、阅读代码就明白通信道理,注释详细; 5、VS2015编译通过。

在网络编程中,Socket 通信是非常重要的一部分。今天咱们就来聊聊如何用 C# 实现 Socket 的异步、同步通信,并且包含断线重连功能,服务端还得支持多个客户端同时连接。整个代码基于 VS2015 编译通过。

服务端代码及分析

同步通信服务端

using System; using System.Net; using System.Net.Sockets; using System.Text; class SyncServer { private TcpListener _tcpListener; private const int BufferSize = 1024; public SyncServer() { // 初始化TcpListener,监听本地11000端口 _tcpListener = new TcpListener(IPAddress.Any, 11000); _tcpListener.Start(); Console.WriteLine("同步服务端已启动,等待客户端连接..."); } public void StartAccepting() { while (true) { // 同步接受客户端连接,这里会阻塞,直到有客户端连接进来 TcpClient client = _tcpListener.AcceptTcpClient(); Console.WriteLine("有新客户端连接:" + ((IPEndPoint)client.Client.RemoteEndPoint).Address); // 处理客户端通信 HandleClient(client); } } private void HandleClient(TcpClient client) { NetworkStream stream = client.GetStream(); byte[] buffer = new byte[BufferSize]; int bytesRead; // 从客户端读取数据,这里也是同步操作,会阻塞 while ((bytesRead = stream.Read(buffer, 0, buffer.Length))!= 0) { string message = Encoding.UTF8.GetString(buffer, 0, bytesRead); Console.WriteLine("收到客户端消息:" + message); // 给客户端回消息 string response = "服务端已收到你的消息:" + message; byte[] responseBuffer = Encoding.UTF8.GetBytes(response); stream.Write(responseBuffer, 0, responseBuffer.Length); } client.Close(); } }

异步通信服务端

using System; using System.Net; using System.Net.Sockets; using System.Text; class AsyncServer { private TcpListener _tcpListener; private const int BufferSize = 1024; public AsyncServer() { _tcpListener = new TcpListener(IPAddress.Any, 11001); _tcpListener.Start(); Console.WriteLine("异步服务端已启动,等待客户端连接..."); // 开始异步接受客户端连接 BeginAccept(); } private void BeginAccept() { _tcpListener.BeginAcceptTcpClient(AcceptCallback, null); } private void AcceptCallback(IAsyncResult ar) { TcpClient client = _tcpListener.EndAcceptTcpClient(ar); Console.WriteLine("有新客户端连接:" + ((IPEndPoint)client.Client.RemoteEndPoint).Address); // 开始异步读取客户端数据 NetworkStream stream = client.GetStream(); byte[] buffer = new byte[BufferSize]; stream.BeginRead(buffer, 0, buffer.Length, ReadCallback, new StateObject { Client = client, Stream = stream, Buffer = buffer }); // 继续接受下一个客户端连接 BeginAccept(); } private void ReadCallback(IAsyncResult ar) { StateObject state = (StateObject)ar.AsyncState; TcpClient client = state.Client; NetworkStream stream = state.Stream; byte[] buffer = state.Buffer; int bytesRead = stream.EndRead(ar); if (bytesRead > 0) { string message = Encoding.UTF8.GetString(buffer, 0, bytesRead); Console.WriteLine("收到客户端消息:" + message); // 给客户端回消息 string response = "服务端已收到你的消息:" + message; byte[] responseBuffer = Encoding.UTF8.GetBytes(response); stream.BeginWrite(responseBuffer, 0, responseBuffer.Length, null, null); // 继续读取下一次数据 stream.BeginRead(buffer, 0, buffer.Length, ReadCallback, state); } else { client.Close(); } } private class StateObject { public TcpClient Client { get; set; } public NetworkStream Stream { get; set; } public byte[] Buffer { get; set; } } }

服务端代码分析

  1. 同步服务端:使用TcpListener监听指定端口。AcceptTcpClient方法是同步的,会阻塞线程直到有客户端连接。处理客户端通信也是同步的,Read方法会阻塞等待客户端发送数据。这种方式简单直接,但在处理多个客户端时性能不佳,因为同一时间只能处理一个客户端的请求。
  2. 异步服务端:同样使用TcpListener,但通过BeginAcceptTcpClientEndAcceptTcpClient实现异步接受客户端连接。异步读取数据时,使用BeginReadEndRead,并且通过StateObject类来传递状态信息,这样可以在回调中继续处理客户端数据,不会阻塞主线程,能同时处理多个客户端连接。

客户端代码及分析

同步通信客户端

using System; using System.Net.Sockets; using System.Text; class SyncClient { private TcpClient _tcpClient; private const string ServerIp = "127.0.0.1"; private const int ServerPort = 11000; private const int BufferSize = 1024; public SyncClient() { Connect(); } private void Connect() { while (true) { try { _tcpClient = new TcpClient(ServerIp, ServerPort); Console.WriteLine("已连接到服务端"); break; } catch (SocketException) { Console.WriteLine("连接失败,5秒后重试..."); System.Threading.Thread.Sleep(5000); } } } public void SendMessage(string message) { NetworkStream stream = _tcpClient.GetStream(); byte[] buffer = Encoding.UTF8.GetBytes(message); stream.Write(buffer, 0, buffer.Length); byte[] responseBuffer = new byte[BufferSize]; int bytesRead = stream.Read(responseBuffer, 0, responseBuffer.Length); string response = Encoding.UTF8.GetString(responseBuffer, 0, bytesRead); Console.WriteLine("收到服务端响应:" + response); } }

异步通信客户端

using System; using System.Net.Sockets; using System.Text; class AsyncClient { private TcpClient _tcpClient; private const string ServerIp = "127.0.0.1"; private const int ServerPort = 11001; private const int BufferSize = 1024; public AsyncClient() { Connect(); } private void Connect() { while (true) { try { _tcpClient = new TcpClient(); _tcpClient.BeginConnect(ServerIp, ServerPort, ConnectCallback, _tcpClient); break; } catch (SocketException) { Console.WriteLine("连接失败,5秒后重试..."); System.Threading.Thread.Sleep(5000); } } } private void ConnectCallback(IAsyncResult ar) { TcpClient client = (TcpClient)ar.AsyncState; client.EndConnect(ar); Console.WriteLine("已连接到服务端"); NetworkStream stream = client.GetStream(); byte[] buffer = new byte[BufferSize]; stream.BeginRead(buffer, 0, buffer.Length, ReadCallback, new StateObject { Client = client, Stream = stream, Buffer = buffer }); } public void SendMessage(string message) { NetworkStream stream = _tcpClient.GetStream(); byte[] buffer = Encoding.UTF8.GetBytes(message); stream.BeginWrite(buffer, 0, buffer.Length, null, null); } private void ReadCallback(IAsyncResult ar) { StateObject state = (StateObject)ar.AsyncState; TcpClient client = state.Client; NetworkStream stream = state.Stream; byte[] buffer = state.Buffer; int bytesRead = stream.EndRead(ar); if (bytesRead > 0) { string response = Encoding.UTF8.GetString(buffer, 0, bytesRead); Console.WriteLine("收到服务端响应:" + response); stream.BeginRead(buffer, 0, buffer.Length, ReadCallback, state); } else { client.Close(); } } private class StateObject { public TcpClient Client { get; set; } public NetworkStream Stream { get; set; } public byte[] Buffer { get; set; } } }

客户端代码分析

  1. 同步客户端Connect方法中使用循环尝试连接服务端,如果连接失败就等待 5 秒后重试,实现了断线重连。SendMessage方法同步发送消息并接收服务端响应。
  2. 异步客户端Connect方法通过BeginConnectEndConnect实现异步连接,连接成功后开始异步读取服务端数据。SendMessage方法异步发送消息。断线重连也是通过循环尝试连接实现的。

通过这些代码,咱们就实现了 C# Socket 的异步、同步通信,并且服务端支持多个客户端连接,客户端和服务端都具备断线重连功能,希望对大家理解 Socket 通信有所帮助。

http://www.jsqmd.com/news/217352/

相关文章:

  • 【 n8n解惑】如何在 n8n 里同时操作数据库和调用外部 API,完成混合计算?
  • 排水管网水质监测系统:确保从源头到排放的全流程水质管控
  • 高性价比升降机品牌推荐,让厨房高处空间触手可及的智能解决方案
  • 基于S7 - 200 PLC和组态王打造5种商品自动售货机
  • STM32步进电机:S型与SpTA加减速控制算法探秘
  • P6KE160A单向 TVS瞬态抑制二极管:3000W工业级瞬态屏障,精准守护160V敏感电路
  • 电磁寻迹小车代码全解析:基于STM32C8T6主控
  • 探秘产线MES系统的核心功能
  • 非科班普通人如何转大模型相关岗位拿下大厂 offer
  • 轮毂电机分布式驱动车辆状态估计:EKF 与 UKF 的实战
  • 雷军回应被指营销大师;字节跳动否认造车;传微软即将开启2026第一次大规模裁员,预计波及超万人 | 极客头条
  • Z-Image-Turbo日志审计功能:生成记录追溯与合规性检查
  • 工控机的抗恶劣环境设计:高温、高湿、强电磁,如何做到稳定运行?
  • 基于总变差正则化模型的图像复原
  • Thinkphp的校园快递仓库管理系统的设计与实现
  • Thinkphp的线上订餐骑手配送管理系统的设计与实现
  • Z-Image-Turbo本地运行安全性验证报告
  • Linux PS2 环境变量详解
  • Linux IFS 环境变量详解
  • Thinkphp的招投标系统的设计与实现
  • 三菱FX3U实现五角星绘制:直线插补与多边形绘制的探索
  • Thinkphp的医疗健康管理平台
  • AI出海新趋势:开源翻译镜像助力中小企业全球化布局
  • Thinkphp的在线医疗预约与咨询平台
  • 【BurpSuite 插件开发】实战篇 - 补充章节
  • 【计算机毕业设计案例】基于python卷积神经网络的淡水生物识别
  • 探索之旅:基于.net 6 的多功能自用工具开发
  • 笔记本电脑连接WiFi后不能上网但是qq微信可以正常使用
  • RISE with SAP 加速落地:2027 大限将至,云 ERP 迁移进入冲刺阶段
  • 【计算机毕业设计案例】基于python卷积神经网络CNN的不同瓶子识别