C# Socket 通信时,怎样判断 Socket 双方是否断开连接
参考内容:
最近需要写个网口通信的上位机程序,我负责写客户端,控制多个客户端与多个服务端简历连接进行通讯,关于异常处理一直有问题,写出来做个记录。
建立连接
/// <summary>
/// 与客户端建立连接:若出错,则开辟一个新线程,在新线程里每隔五秒尝试连接一次,连接成功的话跳出循环,加入到列表中
/// </summary>
private void CreateSocketConnection()
{
int countOfServers = dt_ServerInfo.Rows.Count; //dt表中存着服务端的IP地址和端口号
for (int i = 0; i < countOfServers; i++)
{
Socket serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint serverEndPoint = new IPEndPoint(IPAddress.Parse(dt_ServerInfo.Rows[i]["serverIP"].ToString()),
int.Parse(dt_ServerInfo.Rows[i]["serverPort"].ToString()));
try
{
IAsyncResult result = serverSocket.BeginConnect(serverEndPoint, null, null);
result.AsyncWaitHandle.WaitOne(500);
clientsockets.Add(serverSocket);
socketClients.TryAdd(serverSocket.RemoteEndPoint.ToString(), serverSocket);
}
catch (SocketException) //尝试访问套接字时出错
{
Thread thr_connect = new Thread(() =>
{
try
{
for (int j = 0; j < 10; j++)
{
//serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
serverSocket.Connect(serverEndPoint);
DateTime now = DateTime.Now;
while (now.AddSeconds(5) > DateTime.Now) { }
break;
}
clientsockets.Add(serverSocket);
socketClients.TryAdd(serverSocket.RemoteEndPoint.ToString(), serverSocket);
}
catch { }
});
thr_connect.IsBackground = true;
thr_connect.Start();
}
}
}
发送数据
开辟一个线程来专门发送数据:
/*全局变量*/
Thread Thr_Send = null;
volatile bool Stopflag = false; //发送停止标志位 - volatile保证不被优化掉
......
/*初始化*/
Thr_Send = new Thread(new ThreadStart(SendThread));
Thr_Send.IsBackground = true;
Thr_Send.Start();
/*发送线程内容*/
private void SendThread()
{
while (true)
{
if (!Stopflag)
{
try
{
foreach (var item in socketClients)
{
string serverInfo = item.Key.ToString();
DataRow[] drs = dt_ServerInfo.Select($"ServerIp = '{serverInfo.Split(':')[0]}' And ServerPort = '{serverInfo.Split(':')[1]}'");
byte[] sendByte = hexStringToByteArray(drs[0]["Command"].ToString());
try
{
IAsyncResult result = item.Value.BeginSend(sendByte, 0, sendByte.Length, SocketFlags.None, null, null);
result.AsyncWaitHandle.WaitOne(500);
}
catch (SocketException) // 尝试访问套接字时出错
{
IPEndPoint tempEndpoint = (IPEndPoint)item.Value.RemoteEndPoint;
item.Value.Shutdown(SocketShutdown.Both);
item.Value.Disconnect(true);
item.Value.Close();
Socket tempsocket = item.Value;
socketClients.TryRemove(item.Key, out tempsocket);
Thread thr_reconnect = new Thread(() =>
{
try
{
int j = 0;
for (; j < 10; j++)
{
tempsocket.Connect(tempEndpoint);
DateTime now_temp = DateTime.Now;
while (now_temp.AddSeconds(5) > DateTime.Now) { }
break;
}
if (j!=10)
{
socketClients.TryAdd(tempsocket.RemoteEndPoint.ToString(), tempsocket);
}
Thread.CurrentThread.Abort();
}
catch { }
});
thr_reconnect.IsBackground = true;
thr_reconnect.Start();
}
}
DateTime now = DateTime.Now;
while (now.AddSeconds(1) > DateTime.Now) { }
}
catch (Exception ex)
{
throw ex;
}
}
}
}
BeginReceive也是这么写的,就不贴了,我这只处理了客户端程序错误,没有处理上面说的物理问题,我先测测能用不。
心跳检测
这个不会写,不知道对不对
新开了一个线程:
private byte[] GetKeepLiveData()
{
uint dummy = 0;
byte[] inOptionValues = new byte[Marshal.SizeOf(dummy) * 3];
BitConverter.GetBytes((uint)1).CopyTo(inOptionValues, 0);
BitConverter.GetBytes((uint)3000).CopyTo(inOptionValues, Marshal.SizeOf(dummy)); //keep-alive 间隔
BitConverter.GetBytes((uint)500).CopyTo(inOptionValues, Marshal.SizeOf(dummy) * 2); //尝试间隔
return inOptionValues;
}
private void CheckAlive()
{
Thread.Sleep(10000);
while (true)
{
try
{
lock (socketClients)
{
foreach (var item in socketClients)
{
//if (item.Client.Client.Poll(500, System.Net.Sockets.SelectMode.SelectRead) && (item.Client.Client.Available == 0))
if (item.Value.Poll(500, System.Net.Sockets.SelectMode.SelectRead) && item.Value.Available == 0)
{
//MaterialMessageBox.Show("未收到心跳检测回复");
//心跳检测处理
item.Value.Shutdown(SocketShutdown.Both);
item.Value.Disconnect(true);
item.Value.Close();
Socket tempsocket = item.Value;
socketClients.TryRemove(item.Key, out tempsocket);
}
}
}
}
catch (Exception e)
{
MaterialMessageBox.Show(e.ToString());
}
Thread.Sleep(500);
}
}
C# Socket 通信时,怎样判断 Socket 双方是否断开连接的更多相关文章
- socket通信时如何判断当前连接是否断开--select函数,心跳线程,QsocketNotifier监控socket
client与server建立socket连接之后,如果突然关闭server,此时,如果不在客户端close(socket_fd),会有不好的影响: QsocketNotifier监控socket的槽 ...
- centos socket通信时 connect refused 主要是防火墙问题
centos socket通信时 connect refused 主要是防火墙问题,可以关闭防火墙,或者开放程序中的端口
- C#socket通信时,怎样判断socket双方是否断开连接
我在Server端new了一个socket,然后bind,开了一个线程来accept前来连接的client,每接到一个client前来连接就新开一个线程和它进行通信.我把Server端得到的socke ...
- Socket通信时服务端无响应,客户端超时设置
背景:在写一个客户端的socket程序,服务端没有返回消息,客户端一直在等待. 目标:我需要设置一个时间,如果超过这个时间客户端自动断开连接.最好是在服务端实现,客户端对我来说不可控.
- 解决在进行socket通信时,一端输出流OutputStream不关闭,另一端输入流就接收不到数据
输出的数据需要达到一定的量才会向另一端输出,所以在传输数据的末端添加 \r\n 可以保证不管数据量是多少,都立刻传输到另一端.
- Flex通信-与Java实现Socket通信实例
Flex通信-与Java实现Socket通信实例 转自:http://blessht.iteye.com/blog/1136888 博客分类: Flex 环境准备 [服务器端] JDK1.6,“ja ...
- 【转】C# Socket通信编程
https://www.cnblogs.com/dotnet261010/p/6211900.html#undefined 一:什么是SOCKET socket的英文原义是“孔”或“插座”.作为进程通 ...
- 使用 AT 指令进行 Socket 通信
BC26 支持使用 Socket 进行 TCP 和 UDP 协议通信,这两个协议也是 BC26 支持的众多通信协议的基础.本文讲解如何使用这两个协议与服务器端进行通信.在学习这篇文章前,请首先使用AT ...
- 基于TCP与UDP协议的socket通信
基于TCP与UDP协议的socket通信 C/S架构与初识socket 在开始socket介绍之前,得先知道一个Client端/服务端架构,也就是 C/S 架构,互联网中处处充满了 C/S 架构(Cl ...
- .net平台下C#socket通信(中)
上篇.net平台下C#socket通信(上)介绍了socket通信的基本原理及最基本的通信方式.本文在此基础上就socket通信时经常遇到的问题做一个简单总结,都是项目中的一些小问题,拿来此处便于下次 ...
随机推荐
- ZWS物联网云平台为电器设备智能化,提升产品竞争力
从目前的消费趋势来看,家居电器设备的消费发力点在于年轻人.他们基本由"80"."90后"组成,对于消费理念更加的新颖,对于新产品与新观念更加容易接受.不过相对的 ...
- ajax 获取json值
请求后台获取json: {"success":true,"datamap":{"rebackName":"振勋"}} a ...
- day31 1 tomcat介绍与创建web项目 & 2 继承HttpServlet类、配置webxml全局配置文件 & 3 servlet生命周期 & 4 请求对象HttpServletRequest与响应对象HttpServletResponse
Servlet Java Servlet是运行在Web服务器或应用服务器上的程序,作为客户端(Web浏览器或其他HTTP客户端)和服务端(HTTP服务器上的数据库或应用程序)之间的中间层. 使用Ser ...
- 【Java SE】Day09 继承、super、this、抽象类
一.继承 1.概述 多个类具有相同属性和行为,共性抽取到一个类中(父类) 父类更通用,子类更具体 2.继承后的成员变量 本类:this.成员变量名 父类:super.成员变量名 3.继承后的成员方法 ...
- 【Java EE】Day06 JDBC连接池介绍、C3P0连接池实现、Druid连接池实现、JDBCTemplate
一.数据库连接池介绍 1.引入 之前:每次都要获取连接释放连接 现在:连接重复使用 2.概念: 存放数据库连接的容器 3.实现 DataSource接口 三种实现 标准实现 连接池实现 C3P0 Dr ...
- 【Java EE】Day10 JavaScript高级、DOM、BOM、事件
一.简单入门 1.DOM 功能:获取html文档内容 代码:document.getElementById("id值") 功能: 设置属性值 修改标签体内容:xx.innerHtm ...
- 微软宣布 S2C2F 已被 OpenSSF 采用
开源供应链安全对大多数 IT 领导者来说是个日益严峻的挑战,围绕确保开发人员在构建软件时如何使用和管理开源软件 (OSS) 依赖项的稳健策略至关重要.Microsoft 发布安全供应链消费框架 (S2 ...
- 【leetcode】剑指offer04二维数组查找
很巧妙地把矩阵转化为二叉搜索树(不过好像没什用) class Solution { public: bool findNumberIn2DArray(vector<vector<int&g ...
- selenium 输入文本时报InvalidElementStateException: Message: invalid element state
问题: 当定位输入框时,定位到div标签,如:css->[class="delay el-input"],进行输入操作报invalid element state,显示元素状 ...
- CompletableFuture 使用总结
转载请注明出处: 1.Future使用对比 Future表示一个异步计算的结果.它提供了isDone()来检测计算是否已经完成,并且在计算结束后,可以通过get()方法来获取计算结果.在异步计算中,F ...