下面來羅列一下判斷遠端已經(jīng)斷開的方法:
法一:
當recv()返回值小于等于0時,socket連接斷開。但是還需要判斷 errno是否等于 EINTR,如果errno == EINTR 則說明recv函數(shù)是由于程序接收到信號后返回的,socket連接還是正常的,不應close掉socket連接。
法二:
struct tcp_info info;
int len=sizeof(info);
getsockopt(sock, IPPROTO_TCP, TCP_INFO, &info, (socklen_t *)&len);
if((info.tcpi_state==TCP_ESTABLISHED)) 則說明未斷開 else 斷開
err = 0;
len = sizeof(err);
ret = getsockopt(conn->sock, SOL_SOCKET, SO_ERROR, &err, &len);
if (ret) {
// 當前鏈接異常
printf("Fail to get SO_ERROR: %s\n", strerror(errno));
break;
}
if (err != 0) {
// 當前鏈接異常
printf("socket err is %d\n", err); // EPIPE 32
break;
}
法三:
若使用了select等系統(tǒng)函數(shù),若遠端斷開,則select返回1,recv返回0則斷開。其他注意事項同法一。
法四:
int keepAlive = 1; // 開啟keepalive屬性
int keepIdle = 60; // 如該連接在60秒內(nèi)沒有任何數(shù)據(jù)往來,則進行探測
int keepInterval = 5; // 探測時發(fā)包的時間間隔為5 秒
int keepCount = 3; // 探測嘗試的次數(shù).如果第1次探測包就收到響應了,則后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));
設置后,若斷開,則在使用該socket讀寫時立即失敗,并返回ETIMEDOUT錯誤
法五:
自己實現(xiàn)一個心跳檢測,一定時間內(nèi)未收到自定義的心跳包則標記為已斷開。