C#之实现Scoket心跳机制
C#(6) 
版权声明:本文为博主原创文章,转载请注明出处。
TCP网络长连接
建立起一个TCP连接需要经过“三次握手”:
第一次握手:客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确认;
第二次握手:服务器收到syn包,必须确认客户的SYN(ack=j+1),同时自己也发送一个SYN包(syn=k),即SYN+ACK包,此时服务器进入SYN_RECV状态;
第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ack=k+1),此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手。
握手过程中传送的包里不包含数据,三次握手完毕后,客户端与服务器才正式开始传送数据。理想状态下,TCP连接一旦建立,在通信双方中的任何一方主动关闭连接之前,TCP 连接都将被一直保持下去。断开连接时服务器和客户端均可以主动发起断开TCP连接的请求,断开过程需要经过“四次握手”(过程就不细写了,就是服务器和客户端交互,最终确定断开)
什么是心跳
总的来说,心跳包主要也就是用于长连接的保活和断线处理。一般的应用下,判定时间在30-40秒比较不错。如果实在要求高,那就在6-9秒。
怎么发送心跳?
方法1:应用层自己实现的心跳包
由应用程序自己发送心跳包来检测连接是否正常,大致的方法是:服务器在一个 Timer事件中定时 向客户端发送一个短小精悍的数据包,然后启动一个低级别的线程,在该线程中不断检测客户端的回应, 如果在一定时间内没有收到客户端的回应,即认为客户端已经掉线;同样,如果客户端在一定时间内没 有收到服务器的心跳包,则认为连接不可用。
方法2:TCP的KeepAlive保活机制
因为要考虑到一个服务器通常会连接多个客户端,因此由用户在应用层自己实现心跳包,代码较多 且稍显复杂,而利用TCP/IP协议层为内置的KeepAlive功能来实现心跳功能则简单得多。 不论是服务端还是客户端,一方开启KeepAlive功能后,就会自动在规定时间内向对方发送心跳包, 而另一方在收到心跳包后就会自动回复,以告诉对方我仍然在线。 因为开启KeepAlive功能需要消耗额外的宽带和流量,所以TCP协议层默认并不开启KeepAlive功 能,尽管这微不足道,但在按流量计费的环境下增加了费用,另一方面,KeepAlive设置不合理时可能会 因为短暂的网络波动而断开健康的TCP连接。并且,默认的KeepAlive超时需要7,200,000 MilliSeconds, 即2小时,探测次数为5次。对于很多服务端应用程序来说,2小时的空闲时间太长。因此,我们需要手工开启KeepAlive功能并设置合理的KeepAlive参数。
心跳检测步骤:
1客户端每隔一个时间间隔发生一个探测包给服务器
2客户端发包时启动一个超时定时器
3服务器端接收到检测包,应该回应一个包
4如果客户机收到服务器的应答包,则说明服务器正常,删除超时定时器
5如果客户端的超时定时器超时,依然没有收到应答包,则说明服务器挂了
C#实现的一个简单的心跳
- using System;
- using System.Collections.Generic;
- using System.Threading;
- namespace ConsoleApplication1
- {
- // 客户端离线委托
- public delegate void ClientOfflineHandler(ClientInfo client);
- // 客户端上线委托
- public delegate void ClientOnlineHandler(ClientInfo client);
- public class Program
- {
- /// <summary>
- /// 客户端离线提示
- /// </summary>
- /// <param name="clientInfo"></param>
- private static void ClientOffline(ClientInfo clientInfo)
- {
- Console.WriteLine(String.Format("客户端{0}离线,离线时间:\t{1}", clientInfo.ClientID, clientInfo.LastHeartbeatTime));
- }
- /// <summary>
- /// 客户端上线提示
- /// </summary>
- /// <param name="clientInfo"></param>
- private static void ClientOnline(ClientInfo clientInfo)
- {
- Console.WriteLine(String.Format("客户端{0}上线,上线时间:\t{1}", clientInfo.ClientID, clientInfo.LastHeartbeatTime));
- }
- static void Main()
- {
- // 服务端
- Server server = new Server();
- // 服务端离线事件
- server.OnClientOffline += ClientOffline;
- // 服务器上线事件
- server.OnClientOnline += ClientOnline;
- // 开启服务器
- server.Start();
- // 模拟100个客户端
- Dictionary<Int32, Client> dicClient = new Dictionary<Int32, Client>();
- for (Int32 i = 0; i < 100; i++)
- {
- // 这里传入server只是为了方便而已
- Client client = new Client(i + 1, server);
- dicClient.Add(i + 1, client);
- // 开启客户端
- client.Start();
- }
- System.Threading.Thread.Sleep(1000);
- while (true)
- {
- Console.WriteLine("请输入要离线的ClientID,输入0则退出程序:");
- String clientID = Console.ReadLine();
- if (!String.IsNullOrEmpty(clientID))
- {
- Int32 iClientID = 0;
- Int32.TryParse(clientID, out iClientID);
- if (iClientID > 0)
- {
- Client client;
- if (dicClient.TryGetValue(iClientID, out client))
- {
- // 客户端离线
- client.Offline = true;
- }
- }
- else
- {
- return;
- }
- }
- }
- }
- }
- /// <summary>
- /// 服务端
- /// </summary>
- public class Server
- {
- public event ClientOfflineHandler OnClientOffline;
- public event ClientOnlineHandler OnClientOnline;
- private Dictionary<Int32, ClientInfo> _DicClient;
- /// <summary>
- /// 构造函数
- /// </summary>
- public Server()
- {
- _DicClient = new Dictionary<Int32, ClientInfo>(100);
- }
- /// <summary>
- /// 开启服务端
- /// </summary>
- public void Start()
- {
- // 开启扫描离线线程
- Thread t = new Thread(new ThreadStart(ScanOffline));
- t.IsBackground = true;
- t.Start();
- }
- /// <summary>
- /// 扫描离线
- /// </summary>
- private void ScanOffline()
- {
- while (true)
- {
- // 一秒扫描一次
- System.Threading.Thread.Sleep(1000);
- lock (_DicClient)
- {
- foreach (Int32 clientID in _DicClient.Keys)
- {
- ClientInfo clientInfo = _DicClient[clientID];
- // 如果已经离线则不用管
- if (!clientInfo.State)
- {
- continue;
- }
- // 判断最后心跳时间是否大于3秒
- TimeSpan sp = System.DateTime.Now - clientInfo.LastHeartbeatTime;
- if (sp.Seconds >= 3)
- {
- // 离线,触发离线事件
- if (OnClientOffline != null)
- {
- OnClientOffline(clientInfo);
- }
- // 修改状态
- clientInfo.State = false;
- }
- }
- }
- }
- }
- /// <summary>
- /// 接收心跳包
- /// </summary>
- /// <param name="clientID">客户端ID</param>
- public void ReceiveHeartbeat(Int32 clientID)
- {
- lock (_DicClient)
- {
- ClientInfo clientInfo;
- if (_DicClient.TryGetValue(clientID, out clientInfo))
- {
- // 如果客户端已经上线,则更新最后心跳时间
- clientInfo.LastHeartbeatTime = System.DateTime.Now;
- }
- else
- {
- // 客户端不存在,则认为是新上线的
- clientInfo = new ClientInfo();
- clientInfo.ClientID = clientID;
- clientInfo.LastHeartbeatTime = System.DateTime.Now;
- clientInfo.State = true;
- _DicClient.Add(clientID, clientInfo);
- // 触发上线事件
- if (OnClientOnline != null)
- {
- OnClientOnline(clientInfo);
- }
- }
- }
- }
- }
- /// <summary>
- /// 客户端
- /// </summary>
- public class Client
- {
- public Server Server;
- public Int32 ClientID;
- public Boolean Offline;
- /// <summary>
- /// 构造函数
- /// </summary>
- /// <param name="clientID"></param>
- /// <param name="server"></param>
- public Client(Int32 clientID, Server server)
- {
- ClientID = clientID;
- Server = server;
- Offline = false;
- }
- /// <summary>
- /// 开启客户端
- /// </summary>
- public void Start()
- {
- // 开启心跳线程
- Thread t = new Thread(new ThreadStart(Heartbeat));
- t.IsBackground = true;
- t.Start();
- }
- /// <summary>
- /// 向服务器发送心跳包
- /// </summary>
- private void Heartbeat()
- {
- while (!Offline)
- {
- // 向服务端发送心跳包
- Server.ReceiveHeartbeat(ClientID);
- System.Threading.Thread.Sleep(1000);
- }
- }
- }
- /// <summary>
- /// 客户端信息
- /// </summary>
- public class ClientInfo
- {
- // 客户端ID
- public Int32 ClientID;
- // 最后心跳时间
- public DateTime LastHeartbeatTime;
- // 状态
- public Boolean State;
- }
- }
C#之实现Scoket心跳机制的更多相关文章
- uni-app中websocket的使用 断开重连、心跳机制
前言 最近关于H5和APP的开发中使用到了webSocket,由于web/app有时候会出现网络不稳定或者服务端主动断开,这时候导致消息推送不了的情况,需要客户端进行重连.查阅资料后发现了一个心跳机制 ...
- rabbitmq 的心跳机制&应用
官方文档说: If a consumer dies (its channel is closed, connection is closed, or TCP connection is lost) w ...
- zookeeper心跳机制流程梳理
zookeeper心跳机制流程梳理 Processor链Chain protected void setupRequestProcessors() { RequestProcessor finalPr ...
- 一个Socket连接管理池(心跳机制)
一个Socket连接管理池(心跳机制) http://cuisuqiang.iteye.com/blog/1489661
- ESFramework 开发手册(07) -- 掉线与心跳机制(转)
虽然我们前面已经介绍完了ESFramework开发所需掌握的各种基础设施,但是还不够.想要更好地利用ESFramework这一利器,有些背景知识是我们必须要理解的.就像本文介绍的心跳机制,在严峻的In ...
- 判定生死的心跳机制 --ESFramework 4.0 快速上手(07)
在Internet上采用TCP进行通信的系统,都会遇到一个令人头疼的问题,就是"掉线".而"TCP掉线"这个问题远比我们通常所能想象的要复杂的多 -- 网络拓扑 ...
- 转 互联网推送服务原理:长连接+心跳机制(MQTT协议)
http://blog.csdn.net/zhangzeyuaaa/article/details/39028369 目录(?)[-] 无线移动网络的特点 android系统的推送和IOS的推送有什么 ...
- netty心跳机制测试
netty中有比较完善的心跳机制,(在基础server版本基础上[netty基础--基本收发])添加少量代码即可实现对心跳的监测和处理. 1 server端channel中加入心跳处理机制 // Id ...
- Java: server/client 心跳机制实现 示例
心跳机制 心跳机制是定时发送一个自定义的结构体(心跳包),让对方知道自己还活着,以确保连接的有效性的机制. 大部分CS的应用需要心跳机制.心跳机制一般在Server和Client都要实现,两者实现原理 ...
随机推荐
- 生成式模型 VS 判别式模型
1 定义 1.1 生成式模型 生成式模型(Generative Model)会对x和y的联合分布p(x,y)建模,然后通过贝叶斯公式来求得 p(yi|x),然后选取使得p(yi|x) 最大的 yi,即 ...
- sublime text3 key
Sublime Text 3 3126 注册码 第一个测试通过 —– BEGIN LICENSE —– Michael Barnes Single User License EA7E-821385 8 ...
- triplet改进,变种
1.一开始是FaceNet 2.一个重要的改进:image-based, Ding etal. 3.对于样本挑选的改进: 1)hard samples: hard positive 和hard neg ...
- scrapy框架持久化存储
基于终端指令的持久化存储 基于管道的持久化存储 1.基于终端指令的持久化存储 保证爬虫文件的parse方法中有可迭代类型对象(通常为列表or字典)的返回,该返回值可以通过终端指令的形式写入指定格式的文 ...
- 运放积分电路MULTISIM
有些需要反馈回路
- 关于Adaboost——样本抽样的权值的实际意义
看这篇文章的前提:已经看了PRML中的Adaboost的算法流程 看懂下面的内容必须牢牢记住:Adaboost使用的误差函数是指数误差 文章主要目的:理解样本抽样的权值是为什么那样变化的. 得出的结论 ...
- CSS使用方法
CSS行内样式: 在开始标签内添加style样式属性 如:<p style="color:red;">内容</p> CSS内部样式: 内部样式(嵌入样式), ...
- Estimating Linguistic Complexity for Science Texts--paper
http://aclweb.org/anthology/W18-0505 https://sites.google.com/site/nadeemf0755/research/linguistic-c ...
- 实验吧—隐写术——WP之 奇妙的音乐
点击链接下载压缩包,解压后得到:一个图片,一个压缩包 打开图片: 看到海伦.凯勒我们都知道她是一位盲人,而下面黑色和灰色的点点应该就是盲文了,那么我们百度一下对照表 我们将图片里的盲文对照后得到; k ...
- Go Example--for循环
package main import "fmt" func main() { i := 1 //Go循环只有for, 第一种循环方式 for i<=3 { fmt.Prin ...