参考内容:

C#socket通信时,怎样判断socket双方是否断开连接

C#之Socket断线和重连

Socket.Poll 方法

Socket.Blocking 属性

最近需要写个网口通信的上位机程序,我负责写客户端,控制多个客户端与多个服务端简历连接进行通讯,关于异常处理一直有问题,写出来做个记录。

建立连接

/// <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 双方是否断开连接的更多相关文章

  1. socket通信时如何判断当前连接是否断开--select函数,心跳线程,QsocketNotifier监控socket

    client与server建立socket连接之后,如果突然关闭server,此时,如果不在客户端close(socket_fd),会有不好的影响: QsocketNotifier监控socket的槽 ...

  2. centos socket通信时 connect refused 主要是防火墙问题

    centos socket通信时 connect refused 主要是防火墙问题,可以关闭防火墙,或者开放程序中的端口

  3. C#socket通信时,怎样判断socket双方是否断开连接

    我在Server端new了一个socket,然后bind,开了一个线程来accept前来连接的client,每接到一个client前来连接就新开一个线程和它进行通信.我把Server端得到的socke ...

  4. Socket通信时服务端无响应,客户端超时设置

    背景:在写一个客户端的socket程序,服务端没有返回消息,客户端一直在等待. 目标:我需要设置一个时间,如果超过这个时间客户端自动断开连接.最好是在服务端实现,客户端对我来说不可控.

  5. 解决在进行socket通信时,一端输出流OutputStream不关闭,另一端输入流就接收不到数据

    输出的数据需要达到一定的量才会向另一端输出,所以在传输数据的末端添加 \r\n 可以保证不管数据量是多少,都立刻传输到另一端.

  6. Flex通信-与Java实现Socket通信实例

    Flex通信-与Java实现Socket通信实例  转自:http://blessht.iteye.com/blog/1136888 博客分类: Flex 环境准备 [服务器端] JDK1.6,“ja ...

  7. 【转】C# Socket通信编程

    https://www.cnblogs.com/dotnet261010/p/6211900.html#undefined 一:什么是SOCKET socket的英文原义是“孔”或“插座”.作为进程通 ...

  8. 使用 AT 指令进行 Socket 通信

    BC26 支持使用 Socket 进行 TCP 和 UDP 协议通信,这两个协议也是 BC26 支持的众多通信协议的基础.本文讲解如何使用这两个协议与服务器端进行通信.在学习这篇文章前,请首先使用AT ...

  9. 基于TCP与UDP协议的socket通信

    基于TCP与UDP协议的socket通信 C/S架构与初识socket 在开始socket介绍之前,得先知道一个Client端/服务端架构,也就是 C/S 架构,互联网中处处充满了 C/S 架构(Cl ...

  10. .net平台下C#socket通信(中)

    上篇.net平台下C#socket通信(上)介绍了socket通信的基本原理及最基本的通信方式.本文在此基础上就socket通信时经常遇到的问题做一个简单总结,都是项目中的一些小问题,拿来此处便于下次 ...

随机推荐

  1. 重学c#系列——委托和匿名函数[二十五]

    前言 简单介绍一下什么是委托. 正文 以前也写过委托,这次算是重新归档,和新的补充吧. https://www.cnblogs.com/aoximin/p/13940125.html 有些人说委托是函 ...

  2. 痞子衡嵌入式:MCUXpresso IDE下高度灵活的FreeMarker链接文件模板机制

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是MCUXpresso IDE下高度灵活的FreeMarker链接文件模板机制. 痞子衡之前写过一篇文章 <MCUXpresso I ...

  3. 关于CSDN发布博客接口的研究

    前言 其实我之前就有一个想法,实现用 python 代码来发布博客, 因为我个人做了一个发布到 github 博客软件(其实就是实现 git 命令集成,还有markdown的渲染的软件), 如果我弄明 ...

  4. vulnhub靶场之CONTAINME: 1

    准备: 攻击机:虚拟机kali.本机win10. 靶机:CONTAINME: 1,下载地址:https://download.vulnhub.com/containme/THM-ContainMe-v ...

  5. 【Linux】个人笔记本安装Centos并开放22端口供外网连接

    〇.参考资料 一.配置及安装内容 (一)配置 [today]低配笔记本(装centos):4+500G 个人电脑(装三个节点的虚拟机集群环境):8+628G (二)所装软件 Linux系统(可选cen ...

  6. 修改数据时,一直提示"具有 XXX 的 字典管理 已存在(即数据已存在)

    原代码: class DictConfig(models.Model): """ 字典表 """ id = models.AutoField ...

  7. 手写promise解决回调地狱问题

    在介绍promise之前我们先来看一段代码: 根据案例我们可以看出,这段代码可以无限的嵌套下去,但是每嵌套一层,代码的运行就会降低,而解决回调地狱最好的办法就是new promise 一.什么是 pr ...

  8. 在 K8S Volume 中使用 subPath

    使用 subPath 有时,在单个 Pod 中共享卷以供多方使用是很有用的. volumeMounts.subPath 属性可用于指定所引用的卷内的子路径,而不是其根路径. 下面是一个使用同一共享卷的 ...

  9. visualstudio2017 community版本,有点失去信心了,同样两行代码,外观看不出任何区别,但是一个报错

    不多废话,先上代码 注意查看函数fputs_FILE,该函数的两行代码fopen_s是同样的,但事实上: 第一条fopen_s执行起来会报错,但是第二条就不会!!! /* 练习:获取用户键盘输入,写入 ...

  10. Winform DataGridViewTextBoxCell 编辑添加右键菜单,编辑选中文本

    如上是我们使用DataGridView时,编辑单元格右键会出现系统菜单.现在我们添加自己的右键菜单,并可以操作选中文字. DataGridViewTextBoxCell: DataGridViewTe ...