【协议森林】Linux下socket编程KEEPALIVE使用与原理

1.简介

在实际项目开发中,TCP连接关闭主要有两种情况:

  • 1、连接正常关闭,调用close() 、shutdown()连接优雅关闭,send与recv立马返回错误;
  • 2、连接的对端异常关闭,比如拔掉网线网络断掉,突然断电。

面对上述问题,通常都要检测对端是否处于连接中,主要有两种方法:

  • 1、编写心跳包程序,简单的说就是自己的程序加入一条线程,定时向对端发送数据包,查看是否有ACK,根据ACK的返回情况来管理连接。此方法比较通用,一般使用业务层心跳处理,灵活可控,但改变了现有的协议;
  • 2、使用TCP的keepalive机制,UNIX网络编程不推荐使用SO_KEEPALIVE来做心跳检测。
    在这里插入图片描述

2.keepalive机制

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

int keepalive = 1; // 开启keepalive属性
int keepidle = 60; // 如该连接在60秒内没有任何数据往来,则进行探测
int keepinterval = 5; // 探测时发包的时间间隔为5 秒
int keepcount = 3; // 探测尝试的次数.如果第1次探测包就收到响应了,则后2次的不再发.
setsockopt(rs, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepalive , sizeof(keepalive ));
setsockopt(rs, SOL_TCP, TCP_KEEPIDLE, (void*)&keepidle , sizeof(keepidle ));
setsockopt(rs, SOL_TCP, TCP_KEEPINTVL, (void *)&keepinterval , sizeof(keepinterval ));
setsockopt(rs, SOL_TCP, TCP_KEEPCNT, (void *)&keepcount , sizeof(keepcount ));

3.抓包分析

说明:该抓包对应第四部分的代码
在这里插入图片描述
在这里插入图片描述

4.部分代码

/***************************************************************************
function: NetWorkServerInit
Description:创建网络socket监听服务线程
****************************************************************************/
static int NetWorkServerInit(void)
{
    char *SERVERIP = DEFAULT_CONFIG_SERVER_IP;
    //创建TCP套接字
    ListenSockFd = socket(AF_INET, SOCK_STREAM, 0);
    if(ListenSockFd < 0)
    {
        cks_err("Net work create socket fail!");
        return NET_ERR;
    }

    //设置socket选项
    int opt = 1;
    int idel = 5;
    int interl = 10;
    int cnt = 3;
    setsockopt(ListenSockFd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    setsockopt(ListenSockFd, SOL_SOCKET, SO_KEEPALIVE, &opt, sizeof(opt));
    setsockopt(ListenSockFd, SOL_TCP, TCP_KEEPIDLE, &idel, sizeof(idel));
    setsockopt(ListenSockFd, SOL_TCP, TCP_KEEPINTVL, &interl, sizeof(interl));
    setsockopt(ListenSockFd, SOL_TCP, TCP_KEEPCNT, &cnt, sizeof(cnt));

    //填充本地地址
    struct sockaddr_in local;
    local.sin_port = htons((int)DEFAULT_CONFIG_LISTEN_PORT);
    local.sin_family = AF_INET;
    local.sin_addr.s_addr = inet_addr(SERVERIP);

    //绑定
    if(bind(ListenSockFd, (struct sockaddr *)&local, sizeof(local)) < 0)
    {
        cks_err("Net work bind fail!");
        return NET_ERR;
    }

    //监听检测服务器
    if(listen(ListenSockFd, 5) < 0)
    {
        cks_err("Net work listen fail!");
        return NET_ERR;
    }

    return NET_OK;
 }

5.加入讨论

在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值