- 論壇徽章:
- 0
|
tcp是面向連接的,在實(shí)際應(yīng)用中通常都需要檢測(cè)連接是否還可用.如果不可用,可分為:a. 連接的對(duì)端正常關(guān)閉.b. 連接的對(duì)端非正常關(guān)閉,這包括對(duì)端設(shè)備掉電,程序崩潰,網(wǎng)絡(luò)被中斷等.這種情況是不能也無(wú)法通知對(duì)端的,所以連接會(huì)一直存在,浪費(fèi)國(guó)家的資源.tcp協(xié)議棧有個(gè)keepalive的屬性,可以主動(dòng)探測(cè)socket是否可用,不過(guò)這個(gè)屬性的默認(rèn)值很大.全局設(shè)置可更改/etc/sysctl.conf,加上:net.ipv4.tcp_keepalive_intvl = 20net.ipv4.tcp_keepalive_probes = 3net.ipv4.tcp_keepalive_time = 60在程序中設(shè)置如下:
#include sys/socket.h>
#include netinet/in.h>
#include arpa/inet.h>
#include sys/types.h>
#include netinet/tcp.h>
int keepAlive = 1; // 開(kāi)啟keepalive屬性
int keepIdle = 60; // 如該連接在60秒內(nèi)沒(méi)有任何數(shù)據(jù)往來(lái),則進(jìn)行探測(cè)
int keepInterval = 5; // 探測(cè)時(shí)發(fā)包的時(shí)間間隔為5 秒
int keepCount = 3; // 探測(cè)嘗試的次數(shù).如果第1次探測(cè)包就收到響應(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檢測(cè)到對(duì)端socket不再可用時(shí)(不能發(fā)出探測(cè)包,或探測(cè)包沒(méi)有收到ACK的響應(yīng)包),select會(huì)返回socket可讀,并且在recv時(shí)返回-1,同時(shí)置上errno為ETIMEDOUT.
本文來(lái)自ChinaUnix博客,如果查看原文請(qǐng)點(diǎn):http://blog.chinaunix.net/u/870/showart_501020.html |
|