TCP网络长连接
手机能够使用联网功能是因为手机底层实现了TCP/IP协议,可以使手机终端通过无线网络建立TCP连接。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#实现随机洗牌的方法,包括了C#实现随机洗牌的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了C#实现随机洗牌的方法。分享给大家供大家参考。具体实现方法如下: 希望本文所述对大家的C#程序设计有所帮助。
1. 连接神经元「LED面板」模块。 注:模块介绍请参考「LED面板」。 2. 将事件类积木 当绿色旗帜被点击 拖到脚本区。 2. 添加4个外观类积木 LED面板(1)显示图案()持续(1)秒,实现红心跳动效果。 3. 点击绿色旗帜运行程序。 4. 你也可以绘制更多图案,比如星星或雪花。 小技巧: ☛ 在积木上点击右键可以复制脚本 ☛ 在脚本区的空白处点击右键,可以整理所有积木 5. 保存你的程序
本文向大家介绍C++虚函数的实现机制分析,包括了C++虚函数的实现机制分析的使用技巧和注意事项,需要的朋友参考一下 本文针对C++的虚函数的实现机制进行较为深入的分析,具体如下: 1、简单地说,虚函数是通过虚函数表实现的。那么,什么是虚函数表呢? 事实上,如果一个类中含有虚函数,则系统会为这个类分配一个指针成员指向一张虚函数表(vtbl),表中每一项指向一个虚函数的地址,实现上就是一个函数指针的数
本文向大家介绍C#复制和深度复制的实现方法,包括了C#复制和深度复制的实现方法的使用技巧和注意事项,需要的朋友参考一下 深度复制与浅表复制的区别在于,浅表复制只复制值类型的值,而对于实例所包含的对象依然指向原有实例。 运行结果: 一、List<T>对象中的T是值类型的情况(int 类型等) 对于值类型的List直接用以下方法就可以复制: 二、List<T>对象中的T是引用类型的情况(例如自定义的实
本文向大家介绍Python实现Event回调机制的方法,包括了Python实现Event回调机制的方法的使用技巧和注意事项,需要的朋友参考一下 0.背景 在游戏的UI中,往往会出现这样的情况: 在某个战斗副本中获得了某个道具A,那么当进入主界面的时候,你会看到你的背包UI上有个小红点(意思是有新道具),点击进入背包后,发现新增了道具A,显示个数为1,并且在下个界面中有个使用的按钮由灰色不可使用变成
本文向大家介绍token 机制和实现方式,包括了token 机制和实现方式的使用技巧和注意事项,需要的朋友参考一下 前言 之前在面试的时候被问到过刷新 token 的问题,其实我对 token 验证机制的细节一直不清楚。新项目和后端的同学商量后使用刷新 token 来实现。本文主要分享一下对 token 机制的理解和实现方式。 登录验证的方式 登录验证一般来说有两个目的,一个是为了安全,一个是为了
本文向大家介绍asp.net实现C#绘制太极图的方法,包括了asp.net实现C#绘制太极图的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了asp.net实现C#绘制太极图的方法。分享给大家供大家参考。具体如下: 成品图如下所示: html页面: 注意设置: 后台代码: 希望本文所述对大家的C#程序设计有所帮助。
本文向大家介绍C++实现raw_input的方法,包括了C++实现raw_input的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了C++实现raw_input的方法,分享给大家供大家参考。具体方法分析如下: 用惯了Python,现在写C++的代码感觉有点不太顺畅。今天就来实例演示一下C++实现raw_input的方法。 用过Python的朋友知道,Python中有个raw_inpu