不要使用TCP的KeepAlive功能

TCP的KeepAlive机制用来检测连接是否存活,它并不是TCP协议规范的一部分,但在几乎所有的TCP/IP协议栈中,都实现了KeepAlive功能。
KeepAlive有三个参数:

keepalive_time:探测的超时
keepalive_probes:探测次数
keepalive_intvl:探测间隔

对一个已经建立的tcp连接,如果在keepalive_time时间内双方没有任何的数据包传输,则开启keepalive功能的一端将发送 keepalive数据包,若没有收到应答,则每隔keepalive_intvl时间再发送该数据包,发送keepalive_probes次。一直没有收到应答,则发送rst包关闭连接。若收到应答,则将计时器清零。

但实际发现用处不大,它有几个缺陷:
keepalive只能检测连接是否存活,不能检测连接是否可用。
如果TCP连接中的另一方突然断网,我们并不知道连接断开,此时发送数据失败会进行重传,由于重传包的优先级要高于keepalive的数据包,因此keepalive的数据包无法发送出去。

《TCP/IP详解 卷1:协议》“第21章 TCP的超时与重传”中的例子中:

首次分组传输与复位信号传输之间的时间差约为9分钟,该时间在目前的TCP实现中是不可变的。
对于大多数实现而言,这个总时间是不可调整的。

参考
UNIX网络编程——socket的keep-alive(超时重传)
TCP中已有SO_KEEPALIVE选项,为什么还要在应用层加入心跳包机制??
第21章 TCP的超时与重传

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在LWIP中,使用TCP自带的KEEPALIVE功能需要进行如下设置: 1. 在lwipopts.h文件中开启TCP_KEEPALIVE选项: ``` #define LWIP_TCP_KEEPALIVE 1 ``` 2. 在TCP连接建立后,通过tcp_keepalive()函数设置KEEPALIVE选项: ```c /* 开启KEEPALIVE,时间为30秒,探测间隔为5秒 */ tcp_keepalive(pcb); tcp_keepalive_intvl(pcb, 5000); /* 探测间隔为5秒 */ tcp_keepalive_idle(pcb, 30000); /* 空闲时间为30秒 */ ``` 下面是一个简单的TCP服务器代码实现,包含了KEEPALIVE选项的设置: ```c #include "lwip/tcp.h" /* 定义TCP服务器IP地址和端口号 */ #define SERVER_IP_ADDR "192.168.1.10" #define SERVER_PORT 5001 /* 处理TCP连接请求 */ static err_t tcp_server_accept(void *arg, struct tcp_pcb *newpcb, err_t err) { LWIP_UNUSED_ARG(arg); LWIP_UNUSED_ARG(err); /* 设置KEEPALIVE选项 */ tcp_keepalive(newpcb); tcp_keepalive_intvl(newpcb, 5000); /* 探测间隔为5秒 */ tcp_keepalive_idle(newpcb, 30000); /* 空闲时间为30秒 */ /* 设置接收数据回调函数 */ tcp_recv(newpcb, tcp_server_recv); return ERR_OK; } /* 处理接收到的TCP数据 */ static err_t tcp_server_recv(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err) { LWIP_UNUSED_ARG(arg); LWIP_UNUSED_ARG(err); if (p == NULL) { /* 对方已经关闭连接 */ tcp_close(tpcb); return ERR_OK; } /* 处理接收到的数据 */ ... /* 释放接收缓冲区 */ tcp_recved(tpcb, p->tot_len); /* 释放数据缓冲区 */ pbuf_free(p); return ERR_OK; } int main() { struct tcp_pcb *tcp_server_pcb; /* 创建TCP服务器 */ tcp_server_pcb = tcp_new(); if (tcp_server_pcb == NULL) { /* 创建失败 */ return -1; } /* 绑定IP地址和端口号 */ tcp_bind(tcp_server_pcb, IP_ADDR_ANY, SERVER_PORT); /* 开始监听 */ tcp_listen(tcp_server_pcb); /* 设置接收连接回调函数 */ tcp_accept(tcp_server_pcb, tcp_server_accept); while (1) { /* 处理TCP连接 */ tcp_tmr(); } return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值