Tcp是面向連接的,在實(shí)際應(yīng)用中通常都需要檢測連接是否還可用.如果不可用,可分為:
a. 連接的對端正常關(guān)閉.
b. 連接的對端非正常關(guān)閉,這包括對端設(shè)備掉電,程序崩潰,網(wǎng)絡(luò)被中斷等.這種情況是不能也無法通知對端的,所以連接會一直存在,浪費(fèi)國家的資源.
tcp協(xié)議棧有個(gè)keepalive的屬性,可以主動探測socket是否可用,不過這個(gè)屬性的默認(rèn)值很大.
全局設(shè)置可更改/etc/sysctl.conf,加上:
net.ipv4.tcp_keepalive_intvl = 20
net.ipv4.tcp_keepalive_probes = 3
net.ipv4.tcp_keepalive_time = 60
在程序中設(shè)置如下:
#include
#include
#include
#include
#include
int keepAlive = 1; // 開啟keepalive屬性
int keepIdle = 60; // 如該連接在60秒內(nèi)沒有任何數(shù)據(jù)往來,則進(jìn)行探測
int keepInterval = 5; // 探測時(shí)發(fā)包的時(shí)間間隔為5 秒
int keepCount = 3; // 探測嘗試的次數(shù).如果第1次探測包就收到響應(yīng)了,則后2次的不再發(fā).
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));
在程序中表現(xiàn)為,當(dāng)tcp檢測到對端socket不再可用時(shí)(不能發(fā)出探測包,或探測包沒有收到ACK的響應(yīng)包),select會返回socket可讀,并且在recv時(shí)返回-1,同時(shí)置上errno為ETIMEDOUT。
資料引用:http://www.knowsky.com/395880.html