TCP协议中有长连接和短连接之分。短连接在数据包发送完成后就会自己断开,长连接在发包完毕后,会在一定的时间内保持连接,即我们通常所说的Keepalive(存活定时器)功能。
默认的Keepalive超时需要7,200,000 milliseconds,即2小时,探测次数为5次。它的功效和用户自己实现的心跳机制是一样的。 开启Keepalive功能需要消耗额外的宽带和流量,尽管这微不足道,但在按流量计费的环境下增加了费用,另一方面,Keepalive设置不合理时可能会因为短暂的网络波动而断开健康的TCP连接。

keepalive并不是TCP规范的一部分。在Host Requirements RFC罗列有不使用它的三个理由:(1)在短暂的故障期间,它们可能引起一个良好连接(good connection)被释放(dropped),(2)它们消费了不必要的宽带,(3)在以数据包计费的互联网上它们(额外)花费金钱。然而,在许多的实现中提供了存活定时器。

一些服务器应用程序可能代表客户端占用资源,它们需要知道客户端主机是否崩溃。存活定时器可以为这些应用程序提供探测服务。Telnet服务器和Rlogin服务器的许多版本都默认提供存活选项。

个人计算机用户使用TCP/IP协议通过Telnet登录一台主机,这是能够说明需要使用存活定时器的一个常用例子。如果某个用户在使用结束时只是关掉了电源,而没有注销(log off),那么他就留下了一个半打开(half-open)的连接。如果客户端消失,留给了服务器端半打开的连接,并且服务器又在等待客户端的数据,那么等待将永远持续下去。存活特征的目的就是在服务器端检测这种半打开连接。

也可以在客户端设置存活器选项,且没有不允许这样做的理由,但通常设置在服务器。如果连接两端都需要探测对方是否消失,那么就可以在两端同时设置(比如NFS)。

keepalive工作原理:

若在一个给定连接上,两小时之内无任何活动,服务器便向客户端发送一个探测段。(我们将在下面的例子中看到探测段的样子。)客户端主机必须是下列四种状态之一:

1) 客户端主机依旧活跃(up)运行,并且从服务器可到达。从客户端TCP的正常响应,服务器知道对方仍然活跃。服务器的TCP为接下来的两小时复位存活定时器,如果在这两个小时到期之前,连接上发生应用程序的通信,则定时器重新为往下的两小时复位,并且接着交换数据。

2) 客户端已经崩溃,或者已经关闭(down),或者正在重启过程中。在这两种情况下,它的TCP都不会响应。服务器没有收到对其发出探测的响应,并且在75秒之后超时。服务器将总共发送10个这样的探测,每个探测75秒。如果没有收到一个响应,它就认为客户端主机已经关闭并终止连接。

3) 客户端曾经崩溃,但已经重启。这种情况下,服务器将会收到对其存活探测的响应,但该响应是一个复位,从而引起服务器对连接的终止。

4) 客户端主机活跃运行,但从服务器不可到达。这与状态2类似,因为TCP无法区别它们两个。它所能表明的仅是未收到对其探测的回复。

服务器不必担心客户端主机被关闭然后重启的情况(这里指的是操作员执行的正常关闭,而不是主机的崩溃)。当系统被操作员关闭时,所有的应用程序进程(也就是客户端进程)都将被终止,客户端TCP会在连接上发送一个FIN。收到这个FIN后,服务器TCP向服务器进程报告一个文件结束,以允许服务器检测这种状态。

在第一种状态下,服务器应用程序不知道存活探测是否发生。凡事都是由TCP层处理的,存活探测对应用程序透明,直到后面2,3,4三种状态发生。在这三种状态下,通过服务器的TCP,返回给服务器应用程序错误信息。(通常服务器向网络发出一个读请求,等待客户端的数据。如果存活特征返回一个错误信息,则将该信息作为读操作的返回值返回给服务器。)在状态2,错误信息类似于“连接超时”。状态3则为“连接被对方复位”。第四种状态看起来像连接超时,或者根据是否收到与该连接相关的ICMP错误信息,而可能返回其它的错误信息。

linux内核包含对keepalive的支持。其中使用了三个参数:tcp_keepalive_time(开启keepalive的闲置时 长)tcp_keepalive_intvl(keepalive探测包的发送间隔)和tcp_keepalive_probes (如果对方不予应答,探测包的发送次数);在liunx中,keepalive是一个开关选项,可以通过函数来使能。具体地说,可以使用以下代码:
setsockopt(rs, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepAlive, sizeof(keepAlive));

当tcp检测到对端socket不再可用时(不能发出探测包,或探测包没有收到ACK的响应包),select会返回socket可读,并且在recv时返回-1,同时置上errno为ETIMEDOUT。此时TCP的状态是断开的。

keepalive 参数设置代码如下:

// 开启KeepAlive
BOOL bKeepAlive = TRUE;
int nRet = ::setsockopt(socket_handle, SOL_SOCKET, SO_KEEPALIVE, (char*)&bKeepAlive, sizeof(bKeepAlive));
if (nRet == SOCKET_ERROR)
{
return FALSE;
}

// 设置KeepAlive参数
tcp_keepalive alive_in                = {0};
tcp_keepalive alive_out                = {0};
alive_in.keepalivetime                = 5000;                // 开始首次KeepAlive探测前的TCP空闭时间
alive_in.keepaliveinterval        = 1000;                // 两次KeepAlive探测间的时间间隔
alive_in.onoff                                = TRUE;
unsigned long ulBytesReturn = 0;
nRet = WSAIoctl(socket_handle, SIO_KEEPALIVE_VALS, &alive_in, sizeof(alive_in),
&alive_out, sizeof(alive_out), &ulBytesReturn, NULL, NULL);
if (nRet == SOCKET_ERROR)
{
return FALSE;
}

开启Keepalive选项之后,对于使用IOCP模型的服务器端程序来说,一旦检测到连接断开,GetQueuedCompletionStatus函数将立即返回FALSE,使得服务器端能及时清除该连接、释放该连接相关的资源。对于使用select模型的客户端来说,连接断开被探测到时,以recv目的阻塞在socket上的select方法将立即返回SOCKET_ERROR,从而得知连接已失效,客户端程序便有机会及时执行清除工作、提醒用户或重新连接。

TCP连接非正常断开的检测(KeepAlive探测)

此处的”非正常断开”指TCP连接不是以优雅的方式断开,如网线故障等物理链路的原因,还有突然主机断电等原因

有两种方法可以检测:1.TCP连接双方定时发握手消息 2.利用TCP协议栈中的KeepAlive探测

第二种方法简单可靠,只需对TCP连接两个Socket设定KeepAlive探测。

在windows下使用,要包含MSTcpIP.h的头文件。点击下面的链接即可下载这个文件
   MSTcpIP

备注:长连接虽好,但是比较好用但是占用系统资源比较大。个人建议如无特殊需要,用自己的心跳包机制最好

长连接和Keepalive相关推荐

  1. 谈谈Http长连接和Keep-Alive以及Tcp的Keepalive

    一次性说清楚秒验(本机号码一键登录) 2021-03-02 技术总监面试,提问:Redis热点key解决方案 2021-02-28 [性能测试]轻量级压测工具Hey 2021-02-23 我们知道Ht ...

  2. TCP长连接和Keepalive详解

    http://blog.csdn.net/huangjin0507/article/details/52396580 TCP协议中有长连接和短连接之分.短连接在数据包发送完成后就会自己断开,长连接在发 ...

  3. 理解TCP长连接(Keepalive)

    TCP Keepalive的起源 TCP协议中有长连接和短连接之分.短连接环境下,数据交互完毕后,主动释放连接: 长连接的环境下,进行一次数据交互后,很长一段时间内无数据交互时,客户端可能意外断电.死 ...

  4. C# Socket 长连接设置Keepalive

    首先TCP 底层自带KeepAlive 连接监测机制,通常在指定时间:keepalivetime(毫秒)内没有数据交互,则按keepaliveinterval(毫秒)设定重复发送keep-alive包 ...

  5. 长连接和Keepalive详解

    客户端主机依旧活跃(up)运行,并且从服务器可到达.从客户端TCP的正常响应,服务器知道对方仍然活跃.服务器的TCP为接下来的两小时复位存活定时器,如果在这两个小时到期之前,连接上发生应用程序的通信, ...

  6. WebSocket与http长连接的区别

    前言 事先说明这是鄙人看了一点入门文章总结的粗浅的知识, 并不保证理解完全正确,用来给自己知识梳理用. websocket连接过程概述 WebSocket 建立连接需要先通过一个 http 请求进行和 ...

  7. 计算机网络—URL到显示的过程、Http、Https、长连接短连接

    1.在浏览器中输⼊ URL 地址到显示主页的过程? 1.浏览器解析URL URL = 协议 + 存放资源的主机域名 + 文件路径名 如果没有文件路径名,则访问index.html.default.ht ...

  8. TCP长连接,心跳机制介绍

    TCP长连接,心跳机制介绍 长连接 为何要长连接 心跳 心跳为何设置在服务器端 心跳维持长连接 TCP keep-alive的三个参数 参数的具体意义 心跳的使用场景 长连接 TCP经过三次握手建立连 ...

  9. HTTP 长连接 与 短连接

    最近看了不少博客了解http协议,记录一下长连接与短连接. HTTP的长连接和短连接本质上是TCP长连接和短连接. 短连接: 每次请求新建立一个tcp连接,请求结果返回后立马关闭连接,客户端和服务端都 ...

最新文章

  1. 【机器学习】激活函数(Activation Function)
  2. house robber java_java之 ------------[LeetCode] House Robber 打家劫舍||
  3. Linux 下文件打包和压缩
  4. Linux操作系统Ubuntu部署Oracle篇
  5. 【HDOJ】2577 How to Type
  6. svm 的提前处理
  7. 关于Increased rates of convergence through learning rate adaptation一文的理解
  8. 最简单的卸载vrv方法_低端赤道仪粗对极轴最简单方法
  9. 开发人员如何学习 Kubernetes
  10. Python中转换角度为弧度的radians()方法
  11. POLYCOM RMX2000 激活与初始化 信令板卡初始化
  12. C#LeetCode刷题,走进Google,走近人生
  13. [spark]Spark2.4.6用bulkload写入Hbase1.3.1表的多列
  14. GO语言开发天天生鲜项目第四天 商品后台管理
  15. android搜狗离线安装包,搜狗翻译离线版下载
  16. 电脑使用速度变慢的七大常见原因
  17. java实现计算器_java高级计算器的实现
  18. 如何获取最新的EI期刊杂志目录
  19. 在vs2019上配置opencv,百分百成功
  20. 67 Pod生命周期

热门文章

  1. RocketMQ的Producer详解之分布式事务消息(代码实现以及过程分析)
  2. IDEA与tomcat相关配置
  3. mybatis和ehcache整合
  4. spring项目搭建-导包对象准备
  5. std和android空间,ANDROID平台通讯中STDM加密技术的应用
  6. Spring AbstractBeanFactory
  7. 210221阶段三线程、信号量、互斥锁
  8. STM32 关于外部中断线、中断源和中断服务函数的问题
  9. 调用远程service aidl接口定义
  10. JavaScript语法详解(三)