socket tcp断线重连
一、网上常用方法
1、当Socket.Conneted == false时,调用如下函数进行判断
点击(此处)折叠或打开
- /// <summary>
- /// 当socket.connected为false时,进一步确定下当前连接状态
- /// </summary>
- /// <returns></returns>
- private bool IsSocketConnected()
- {
- #region remarks
- /********************************************************************************************
- * 当Socket.Conneted为false时, 如果您需要确定连接的当前状态,请进行非阻塞、零字节的 Send 调用。
- * 如果该调用成功返回或引发 WAEWOULDBLOCK 错误代码 (10035),则该套接字仍然处于连接状态;
- * 否则,该套接字不再处于连接状态。
- * Depending on http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.connected.aspx?cs-save-lang=1&cs-lang=csharp#code-snippet-2
- ********************************************************************************************/
- #endregion
- #region 过程
- // This is how you can determine whether a socket is still connected.
- bool connectState = true;
- bool blockingState = socket.Blocking;
- try
- {
- byte[] tmp = new byte[1];
- socket.Blocking = false;
- socket.Send(tmp, 0, 0);
- //Console.WriteLine("Connected!");
- connectState = true; //若Send错误会跳去执行catch体,而不会执行其try体里其之后的代码
- }
- catch (SocketException e)
- {
- // 10035 == WSAEWOULDBLOCK
- if (e.NativeErrorCode.Equals(10035))
- {
- //Console.WriteLine("Still Connected, but the Send would block");
- connectState = true;
- }
- else
- {
- //Console.WriteLine("Disconnected: error code {0}!", e.NativeErrorCode);
- connectState = false;
- }
- }
- finally
- {
- socket.Blocking = blockingState;
- }
- //Console.WriteLine("Connected: {0}", client.Connected);
- return connectState;
- #endregion
- }
2、根据socket.poll判断
点击(此处)折叠或打开
- /// <summary>
- /// 另一种判断connected的方法,但未检测对端网线断开或ungraceful的情况
- /// </summary>
- /// <param name="s"></param>
- /// <returns></returns>
- static bool IsSocketConnected(Socket s)
- {
- #region remarks
- /* As zendar wrote, it is nice to use the Socket.Poll and Socket.Available, but you need to take into conside ration
- * that the socket might not have been initialized in the first place.
- * This is the last (I believe) piece of information and it is supplied by the Socket.Connected property.
- * The revised version of the method would looks something like this:
- * from:http://stackoverflow.com/questions/2661764/how-to-check-if-a- socket-is-connected-disconnected-in-c */
- #endregion
- #region 过程
- return !((s.Poll(1000, SelectMode.SelectRead) && (s.Available == 0)) || !s.Connected);
- /* The long, but simpler-to-understand version:
- bool part1 = s.Poll(1000, SelectMode.SelectRead);
- bool part2 = (s.Available == 0);
- if ((part1 && part2 ) || !s.Connected)
- return false;
- else
- return true;
- */
- #endregion
- }
总结:--1--此两种方法出处可在函数体中的remark中找到链接
--2--此两种方法适用于对端正常关闭socket下的本地socket状态检测,在非正常关闭如断电、拔网线的情况下不起作用
因为Socket.Conneted存在bug,详见.Net Bugs
二、支持物理断线重连功能的类
利用BeginReceive + KeepAlive实现物理断线重连,初步测验了一下,正常。(部分代码参考帖子#26及blog在C#中利用keep-alive处理socket网络异常断开)
Keep-Alive机制的介绍请看TCP Keepalive HOWTO
以此备忘,同时希望能帮助到有需要的同学。
点击(此处)折叠或打开
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- using System.Net.Sockets;
- using System.Net;
- using System.Threading;
- namespace MySocket
- {
- public class Socket_wrapper
- {
- //委托
- private delegate void delSocketDataArrival(byte[] data);
- static delSocketDataArrival socketDataArrival = socketDataArrivalHandler;
- private delegate void delSocketDisconnected();
- static delSocketDisconnected socketDisconnected = socketDisconnectedHandler;
- public static Socket theSocket = null;
- private static string remoteHost = "192.168.1.71";
- private static int remotePort = 6666;
- private static String SockErrorStr = null;
- private static ManualResetEvent TimeoutObject = new ManualResetEvent(false);
- private static Boolean IsconnectSuccess = false; //异步连接情况,由异步连接回调函数置位
- private static object lockObj_IsConnectSuccess = new object();
- /// <summary>
- /// 构造函数
- /// </summary>
- /// <param name="strIp"></param>
- /// <param name="iPort"></param>
- public Socket_wrapper(string strIp, int iPort)
- {
- remoteHost = strIp;
- remotePort = iPort;
- }
- /// <summary>
- /// 设置心跳
- /// </summary>
- private static void SetXinTiao()
- {
- //byte[] inValue = new byte[] { 1, 0, 0, 0, 0x20, 0x4e, 0, 0, 0xd0, 0x07, 0, 0 };// 首次探测时间20 秒, 间隔侦测时间2 秒
- byte[] inValue = new byte[] { 1, 0, 0, 0, 0x88, 0x13, 0, 0, 0xd0, 0x07, 0, 0 };// 首次探测时间5 秒, 间隔侦测时间2 秒
- theSocket.IOControl(IOControlCode.KeepAliveValues, inValue, null);
- }
- /// <summary>
- /// 创建套接字+异步连接函数
- /// </summary>
- /// <returns></returns>
- private static bool socket_create_connect()
- {
- IPAddress ipAddress = IPAddress.Parse(remoteHost);
- IPEndPoint remoteEP = new IPEndPoint(ipAddress, remotePort);
- theSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
- theSocket.SendTimeout = 1000;
- SetXinTiao();//设置心跳参数
- #region 异步连接代码
- TimeoutObject.Reset(); //复位timeout事件
- try
- {
- theSocket.BeginConnect(remoteEP, connectedCallback, theSocket);
- }
- catch (Exception err)
- {
- SockErrorStr = err.ToString();
- return false;
- }
- if (TimeoutObject.WaitOne(10000, false))//直到timeout,或者TimeoutObject.set()
- {
- if (IsconnectSuccess)
- {
- return true;
- }
- else
- {
- return false;
- }
- }
- else
- {
- SockErrorStr = "Time Out";
- return false;
- }
- #endregion
- }
- /// <summary>
- /// 同步receive函数
- /// </summary>
- /// <param name="readBuffer"></param>
- /// <returns></returns>
- public string socket_receive(byte[] readBuffer)
- {
- try
- {
- if (theSocket == null)
- {
- socket_create_connect();
- }
- else if (!theSocket.Connected)
- {
- if (!IsSocketConnected())
- Reconnect();
- }
- int bytesRec = theSocket.Receive(readBuffer);
- if (bytesRec == 0)
- {
- //warning 0 bytes received
- }
- return Encoding.ASCII.GetString(readBuffer, 0, bytesRec);
- }
- catch (SocketException se)
- {
- //print se.ErrorCode
- throw;
- }
- }
- /// <summary>
- /// 同步send函数
- /// </summary>
- /// <param name="sendMessage"></param>
- /// <returns></returns>
- public bool socket_send(string sendMessage)
- {
- if (checkSocketState())
- {
- return SendData(sendMessage);
- }
- return false;
- }
- /// <summary>
- /// 断线重连函数
- /// </summary>
- /// <returns></returns>
- private static bool Reconnect()
- {
- //关闭socket
- theSocket.Shutdown(SocketShutdown.Both);
- theSocket.Disconnect(true);
- IsconnectSuccess = false;
- theSocket.Close();
- //创建socket
- return socket_create_connect();
- }
- /// <summary>
- /// 当socket.connected为false时,进一步确定下当前连接状态
- /// </summary>
- /// <returns></returns>
- private bool IsSocketConnected()
- {
- #region remarks
- /********************************************************************************************
- * 当Socket.Conneted为false时, 如果您需要确定连接的当前状态,请进行非阻塞、零字节的 Send 调用。
- * 如果该调用成功返回或引发 WAEWOULDBLOCK 错误代码 (10035),则该套接字仍然处于连接状态;
- * 否则,该套接字不再处于连接状态。
- * Depending on http://msdn.microsoft.com/zh-cn/library/system.net.sockets.socket.connected.aspx?cs-save-lang=1&cs-lang=csharp#code-snippet-2
- ********************************************************************************************/
- #endregion
- #region 过程
- // This is how you can determine whether a socket is still connected.
- bool connectState = true;
- bool blockingState = theSocket.Blocking;
- try
- {
- byte[] tmp = new byte[1];
- theSocket.Blocking = false;
- theSocket.Send(tmp, 0, 0);
- //Console.WriteLine("Connected!");
- connectState = true; //若Send错误会跳去执行catch体,而不会执行其try体里其之后的代码
- }
- catch (SocketException e)
- {
- // 10035 == WSAEWOULDBLOCK
- if (e.NativeErrorCode.Equals(10035))
- {
- //Console.WriteLine("Still Connected, but the Send would block");
- connectState = true;
- }
- else
- {
- //Console.WriteLine("Disconnected: error code {0}!", e.NativeErrorCode);
- connectState = false;
- }
- }
- finally
- {
- theSocket.Blocking = blockingState;
- }
- //Console.WriteLine("Connected: {0}", client.Connected);
- return connectState;
- #endregion
- }
- /// <summary>
- /// 另一种判断connected的方法,但未检测对端网线断开或ungraceful的情况
- /// </summary>
- /// <param name="s"></param>
- /// <returns></returns>
- public static bool IsSocketConnected(Socket s)
- {
- #region remarks
- /* As zendar wrote, it is nice to use the Socket.Poll and Socket.Available, but you need to take into consideration
- * that the socket might not have been initialized in the first place.
- * This is the last (I believe) piece of information and it is supplied by the Socket.Connected property.
- * The revised version of the method would looks something like this:
- * from:http://stackoverflow.com/questions/2661764/how-to-check-if-a- socket-is-connected-disconnected-in-c */
- #endregion
- #region 过程
- if (s == null)
- return false;
- return !((s.Poll(1000, SelectMode.SelectRead) && (s.Available == 0)) || !s.Connected);
- /* The long, but simpler-to-understand version:
- bool part1 = s.Poll(1000, SelectMode.SelectRead);
- bool part2 = (s.Available == 0);
- if ((part1 && part2 ) || !s.Connected)
- return false;
- else
- return true;
- */
- #endregion
- }
- /// <summary>
- /// 异步连接回调函数
- /// </summary>
- /// <param name="iar"></param>
- static void connectedCallback(IAsyncResult iar)
- {
- #region <remarks>
- /// 1、置位IsconnectSuccess
- #endregion </remarks>
- lock (lockObj_IsConnectSuccess)
- {
- Socket client = (Socket)iar.AsyncState;
- try
- {
- client.EndConnect(iar);
- IsconnectSuccess = true;
- StartKeepAlive(); //开始KeppAlive检测
- }
- catch (Exception e)
- {
- //Console.WriteLine(e.ToString());
- SockErrorStr = e.ToString();
- IsconnectSuccess = false;
- }
- finally
- {
- TimeoutObject.Set();
- }
- }
- }
- /// <summary>
- /// 开始KeepAlive检测函数
- /// </summary>
- private static void StartKeepAlive()
- {
- theSocket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(OnReceiveCallback), theSocket);
- }
- /// <summary>
- /// BeginReceive回调函数
- /// </summary>
- static byte[] buffer = new byte[1024];
- private static void OnReceiveCallback(IAsyncResult ar)
- {
- try
- {
- Socket peerSock = (Socket)ar.AsyncState;
- int BytesRead = peerSock.EndReceive(ar);
- if (BytesRead > 0)
- {
- byte[] tmp = new byte[BytesRead];
- Array.ConstrainedCopy(buffer, 0, tmp, 0, BytesRead);
- if (socketDataArrival != null)
- {
- socketDataArrival(tmp);
- }
- }
- else//对端gracefully关闭一个连接
- {
- if (theSocket.Connected)//上次socket的状态
- {
- if (socketDisconnected != null)
- {
- //1-重连
- socketDisconnected();
- //2-退出,不再执行BeginReceive
- return;
- }
- }
- }
- //此处buffer似乎要清空--待实现 zq
- theSocket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, new AsyncCallback(OnReceiveCallback), theSocket);
- }
- catch (Exception ex)
- {
- if (socketDisconnected != null)
- {
- socketDisconnected(); //Keepalive检测网线断开引发的异常在这里捕获
- return;
- }
- }
- }
- /// <summary>
- /// 异步收到消息处理器
- /// </summary>
- /// <param name="data"></param>
- private static void socketDataArrivalHandler(byte[] data)
- {
- }
- /// <summary>
- /// socket由于连接中断(软/硬中断)的后续工作处理器
- /// </summary>
- private static void socketDisconnectedHandler()
- {
- Reconnect();
- }
- /// <summary>
- /// 检测socket的状态
- /// </summary>
- /// <returns></returns>
- public static bool checkSocketState()
- {
- try
- {
- if (theSocket == null)
- {
- return socket_create_connect();
- }
- else if (IsconnectSuccess)
- {
- return true;
- }
- else//已创建套接字,但未connected
- {
- #region 异步连接代码
- TimeoutObject.Reset(); //复位timeout事件
- try
- {
- IPAddress ipAddress = IPAddress.Parse(remoteHost);
- IPEndPoint remoteEP = new IPEndPoint(ipAddress, remotePort);
- theSocket.BeginConnect(remoteEP, connectedCallback, theSocket);
- SetXinTiao();//设置心跳参数
- }
- catch (Exception err)
- {
- SockErrorStr = err.ToString();
- return false;
- }
- if (TimeoutObject.WaitOne(2000, false))//直到timeout,或者TimeoutObject.set()
- {
- if (IsconnectSuccess)
- {
- return true;
- }
- else
- {
- return false;
- }
- }
- else
- {
- SockErrorStr = "Time Out";
- return false;
- }
- #endregion
- }
- }
- catch (SocketException se)
- {
- SockErrorStr = se.ToString();
- return false;
- }
- }
- /// <summary>
- /// 同步发送
- /// </summary>
- /// <param name="dataStr"></param>
- /// <returns></returns>
- public static bool SendData(string dataStr)
- {
- bool result = false;
- if (dataStr == null || dataStr.Length < 0)
- return result;
- try
- {
- byte[] cmd = Encoding.Default.GetBytes(dataStr);
- int n = theSocket.Send(cmd);
- if (n < 1)
- result = false;
- }
- catch (Exception ee)
- {
- SockErrorStr = ee.ToString();
- result = false;
- }
- return result;
- }
- }
- }
- 参考
- https://www.cnblogs.com/MRRAOBX/articles/3601856.html
socket tcp断线重连的更多相关文章
- 关于socket tcp 断线重连
这个问题困扰过我几次,都没有来得及研究,今天研究一下. 首先写一个最简易的socket tcp程序,连接成功后再关闭服务器然后再用客户端各种操作看是什么情况 测试表明 (1)客户端已经连接,当服务端关 ...
- Socket实现断线重连
客户端维护一个线程安全的待发送信息队列 开启死循环 判断Socket = null 调用Socket的sendUrgentData(0xFF)发送1个字节的心跳包 捕捉到连接异常后就关 ...
- TCP断线重连
struct sockaddr_in TempSadd; TempSadd.sin_family = AF_INET; TempSadd.sin_port = htons(m_ServerPort); ...
- socket 断线重连
send发送数据时,发送失败,进行如下重连处理: ) < )//serbuf中有数据可以发送才会执行这条语句 { printf("serial to tcp send msg erro ...
- C#之Socket断线重连
一.网上常用方法 1.当Socket.Conneted == false时,调用如下函数进行判断 /// /// 当socket.connected为false时,进一步确定下当前连接状态 /// / ...
- 发现电脑上装着liteide,就用golang做一个TCP通讯测试(支持先启动client端和断线重连)
1.参考https://www.cnblogs.com/yin5th/p/9274495.html server端 main.go package main import ( "fmt&qu ...
- android java socket断线重连
android java socket断线重连 thread = new Thread(new Runnable() { @Override public void run() { while (tr ...
- 基于TCP通信的客户端断线重连
转载:http://www.cnblogs.com/networkcomms/p/4304362.html 源码下载 在CS程序中,断线重连应该是一个常见的功能. 此处的断线重连主要指的是服务器端因为 ...
- netty4 断线重连
转载:http://www.tuicool.com/articles/B7RzMbY 一 实现心跳检测 原理:当服务端每隔一段时间就会向客户端发送心跳包,客户端收到心跳包后同样也会回一个心跳包给服务端 ...
- 浅谈IM软件client的断线重连、心跳和长在线
版权声明:原创文章,未经博主同意禁止转载.欢迎点击头像上方"郭晓东的专栏"查看专栏 https://blog.csdn.net/hherima/article/details/27 ...
随机推荐
- 在JDBC中使用带参数的SQL语句
ADO.Net中,支持带参数的SQL语句,例如:Select * from Tables where column1=@column1,其中@column1为SQL参数,使用起来非常方便,而JDBC中 ...
- Fail pg walkthrough Intermediate
nmap ┌──(root㉿kali)-[/home/ftpuserr] └─# nmap -p- -A 192.168.159.126 Starting Nmap 7.94SVN ( https:/ ...
- grpc的使用
需要保证电脑中安装了:protobuf安装教程 如果出现报错请看博客:protobuf报错问题解决 基本使用demo地址:demo 安全传输.流式传输的demo地址:demo2 简介: rpc微服务, ...
- Windows中安装和配置Maven
1.下载 下载地址:https://maven.apache.org/download.cgi 下载文件:https://dlcdn.apache.org/maven/maven-3/3.9.6/bi ...
- C盘扩展卷碰到的那些事-->不是同一块物理磁盘操作扩展卷是有坑的
自己电脑上面用过win10系统资源管理器扩展卷的功能,用过几次都成功扩容了磁盘空间,简单说一下原理: 就是将剩余未分配的磁盘空间划给要扩展的磁盘. 这天公司的电脑C盘老是红色提示空间不足,那就扩充容量 ...
- Aspire+扣子智能体实现AI自动CodeReview
一.引言 Code Review在软件开发中扮演着至关重要的角色,它不仅能够提升代码质量,确保代码的可维护性和一致性,还能促进团队成员之间的知识共享和技术提升. 传统的代码审查过程面临着诸多挑战和局限 ...
- 发那科FANUC机器人M-710iC减速机维修看这几步
发那科作为全球知名的工业机器人制造商,其M-710iC型号机器人在工业界享有盛誉.然而,即便是最顶尖的设备也难免会遇到维修问题,其中四轴传动齿轮箱的维修尤为关键.本文将深入探讨发那科FANUC机器人 ...
- C# List LinQ Lambda 表达式
------------恢复内容开始------------ # 参考链接 : https://blog.csdn.net/wori/article/details/113144580 首先 => ...
- 使用Go复刻skiplist核心功能
0.引言 正好做LC每日一题要求实现一个跳表,于是学习了redis的扩展skiplist,并使用Go进行复刻学习.学习参考了文章:Redis内部数据结构详解(6)--skiplist - 铁蕾的个人博 ...
- 【最佳方案】RAG 技术深度剖析及 MaxKB 在企业 AI 落地中的应用策略
RAG 已经成为 LLM 大语言模型在企业落地的最佳方案,其中主要是因为 RAG 能够解决幻觉问题.时效性问题以及数据安全问题. 解决幻觉问题: LLM 文本生成的底层原理是基于概率的 token b ...