如何設(shè)置socket的Connect超時(shí)(linux)
今天發(fā)現(xiàn)自己的系統(tǒng)存在很嚴(yán)重缺陷,當(dāng)前臺(tái)關(guān)閉的時(shí)候后臺(tái)就無法正常工作,原因 很好定位,后臺(tái)的socket連接超時(shí)時(shí)間過長,系統(tǒng)默認(rèn)時(shí)間好像是75秒,于是找資料,根據(jù)下邊文章中的內(nèi)容解決了,把超時(shí)時(shí)間設(shè)為5秒后,感覺好多 了。看來還有好多東西需要慢慢挖掘阿!
如何設(shè)置socket的Connect超時(shí)(linux)
[From]
http://dev.cbw.com/c/c/200510195601_4292587.shtml
1. 首先將標(biāo)志位設(shè)為Non-blocking模式,準(zhǔn)備在非阻塞模式下調(diào)用connect函數(shù)
2.調(diào)用connect,正常情況下,因?yàn)門CP三次 握手需要一些時(shí)間;而非阻塞調(diào)用只要不能立即完成就會(huì)返回錯(cuò)誤,所以這里會(huì)返回EINPROGRESS,表示在建立連接但還沒有完成。
3.在讀套 接口描述符集(fd_set rset)和寫套接口描述符集(fd_set wset)中將當(dāng)前套接口置位(用FD_ZERO()、FD_SET()宏),并設(shè)置好超時(shí)時(shí)間(struct timeval *timeout)
4. 調(diào)用select( socket, &rset, &wset, NULL, timeout )
返回0表示connect超 時(shí)
如果你設(shè)置的超時(shí)時(shí)間大于75秒就沒有必要這樣做了,因?yàn)閮?nèi)核中對(duì)connect有超時(shí)限制就是75秒。
[From]
http://www.ycgczj.com.cn/34733.html
網(wǎng) 絡(luò)編程中socket的分量我想大家都很清楚了,socket也就是套接口,在套接口編程中,提到超時(shí)的概念,我們一下子就能想到3個(gè):發(fā)送超時(shí),接收超 時(shí),以及select超時(shí)(注: select函數(shù)并不是只用于套接口的,但是套接口編程中用的比較多),在connect到目標(biāo)主機(jī)的時(shí)候,這個(gè)超時(shí)是不由我們來設(shè)置的。不過正常情況下 這個(gè)超時(shí)都很長,并且connect又是一個(gè)阻塞方法,一個(gè)主機(jī)不能連接,等著connect返回還能忍受,你的程序要是要試圖連接多個(gè)主機(jī),恐怕遇到多 個(gè)不能連接的主機(jī)的時(shí)候,會(huì)塞得你受不了的。我也廢話少說,先說說我的方法,如果你覺得你已掌握這種方法,你就不用再看下去了,如果你還不了解,我愿意與 你分享。本文是已在Linux下的程序?yàn)槔樱贿^拿到Windows中方法也是一樣,無非是換幾個(gè)函數(shù)名字罷了。
Linux中要給 connect設(shè)置超時(shí),應(yīng)該是有兩種方法的。一種是該系統(tǒng)的一些參數(shù),這個(gè)方法我不講,因?yàn)槲抑v不清楚:P,它也不是編程實(shí)現(xiàn)的。另外一種方法就是變相 的實(shí)現(xiàn)connect的超時(shí),我要講的就是這個(gè)方法,原理上是這樣的:
1.建立socket
2.將該socket設(shè)置為非阻塞模式
3. 調(diào)用connect()
4.使用select()檢查該socket描述符是否可寫(注意,是可寫)
5.根據(jù)select()返回的結(jié)果 判斷connect()結(jié)果
6.將socket設(shè)置為阻塞模式(如果你的程序不需要用阻塞模式的,這步就省了,不過一般情況下都是用阻塞模式的, 這樣也容易管理)
如果你對(duì)網(wǎng)絡(luò)編程很熟悉的話,其實(shí)我一說出這個(gè)過程你就知道怎么寫你的程序了,下面給出我寫的一段程序,僅供參考。
/******************************
* Time out for connect()
* Write by Kerl W
******************************/
#include <sys/socket.h>
#include <sys/types.h>
#define TIME_OUT_TIME 20 //connect超時(shí)時(shí)間20秒
int main(int argc , char **argv)
{
………………
int sockfd = socket(AF_INET, SOCK_STREAM, 0);
if(sockfd < 0) exit(1);
struct sockaddr_in serv_addr;
………//以服務(wù)器地址填充結(jié)構(gòu)serv_addr
int error=-1, len;
len = sizeof(int);
timeval tm;
fd_set set;
unsigned long ul = 1;
ioctl(sockfd, FIONBIO, &ul); //設(shè)置為非阻塞模式
bool ret = false;
if( connect(sockfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) == -1)
{
tm.tv_set = TIME_OUT_TIME;
tm.tv_uset = 0;
FD_ZERO(&set);
FD_SET(sockfd, &set);
if( select(sockfd+1, NULL, &set, NULL, &tm) > 0)
{
getsockopt(sockfd, SOL_SOCKET, SO_ERROR, &error, (socklen_t *)&len);
if(error == 0) ret = true;
else ret = false;
} else ret = false;
}
else ret = true;
ul = 0;
ioctl(sockfd, FIONBIO, &ul); //設(shè)置為阻塞模式
if(!ret)
{
close( sockfd );
fprintf(stderr , "Cannot Connect the server!n");
return;
}
fprintf( stderr , "Connected!n");
//下面還可以進(jìn)行發(fā)包收包操作
……………
}
以上代碼片段, 僅供參考,也是為初學(xué)者提供一些提示,主要用到的幾個(gè)函數(shù),select, ioctl, getsockopt都可以找到相關(guān)資料,具體用法我這里就不贅述了,你只需要在linux中輕輕的敲一個(gè)man <函數(shù)名>就能夠看到它的用法。
此外我需要說明的幾點(diǎn)是,雖然我們用ioctl把套接口設(shè)置為非阻塞模式,不過select本身是阻 塞的,阻塞的時(shí)間就是其超時(shí)的時(shí)間由調(diào)用select 的時(shí)候的最后一個(gè)參數(shù)timeval類型的變量指針指向的timeval結(jié)構(gòu)變量來決定的,timeval結(jié)構(gòu)由一個(gè)表示秒數(shù)的和一個(gè)表示微秒數(shù) (long類型)的成員組成,一般我們?cè)O(shè)置了秒數(shù)就行了,把微妙數(shù)設(shè)為0(注:1秒等于100萬微秒)。而select函數(shù)另一個(gè)值得一提的參數(shù)就是上面 我們用到的fd_set類型的變量指針。調(diào)用之前,這個(gè)變量里面存了要用select來檢查的描述符,調(diào)用之后,針對(duì)上面的程序這里面是可寫的描述符,我 們可以用宏FD_ISSET來檢查某個(gè)描述符是否在其中。由于我這里只有一個(gè)套接口描述符,我就沒有使用FD_ISSET宏來檢查調(diào)用select之后這 個(gè)sockfd是否在set里面,其實(shí)是需要加上這個(gè)判斷的。不過我用了getsockopt來檢查,這樣才可以判斷出這個(gè)套接口是否是真的連接上了,因 為我們只是變相的用select來檢查它是否連接上了,實(shí)際上select檢查的是它是否可寫,而對(duì)于可寫,是針對(duì)以下三種條件任一條件滿足時(shí)都表示可寫 的:
1)套接口發(fā)送緩沖區(qū)中的可用控件字節(jié)數(shù)大于等于套接口發(fā)送緩沖區(qū)低潮限度的當(dāng)前值,且或者i)套接口已連接,或者ii)套接口不要求連接 (UDP方式的)
2)連接的寫這一半關(guān)閉。
3)有一個(gè)套接口錯(cuò)誤待處理。
這樣,我們就需要用getsockopt函數(shù)來獲取套接 口目前的一些信息來判斷是否真的是連接上了,沒有連接上的時(shí)候還能給出發(fā)生了什么錯(cuò)誤,當(dāng)然我程序中并沒有標(biāo)出那么多狀態(tài),只是簡單的表示可連接/不可連 接。
下面我來談?wù)剬?duì)這個(gè)程序測(cè)試的結(jié)果。我針對(duì)3種情形做了測(cè)試:
1. 目標(biāo)機(jī)器網(wǎng)絡(luò)正常的情況
可以連接到目標(biāo)主機(jī),并能成功以 阻塞方式進(jìn)行發(fā)包收包作業(yè)。
2. 目標(biāo)機(jī)器網(wǎng)絡(luò)斷開的情況
在等待設(shè)置的超時(shí)時(shí)間(上面的程序中為20秒)后,顯示目標(biāo)主機(jī)不能連接。
3. 程序運(yùn)行前斷開目標(biāo)機(jī)器網(wǎng)絡(luò),超時(shí)時(shí)間內(nèi),恢復(fù)目標(biāo)機(jī)器的網(wǎng)絡(luò)
在恢復(fù)目標(biāo)主機(jī)網(wǎng)絡(luò)連接之前,程序一只等待,恢復(fù)目標(biāo)主機(jī)后,程序顯示連接目標(biāo)主 機(jī)成功,并能成功以阻塞方式進(jìn)行發(fā)包收包作業(yè)。
以上各種情況的測(cè)試結(jié)果表明,這種設(shè)置connect超時(shí)的方法是完全可行的。我自己是把這種設(shè)置 了超時(shí)的connect封裝到了自己的類庫,用在一套監(jiān)控系統(tǒng)中,到目前為止,運(yùn)行還算正常。這種編程實(shí)現(xiàn)的connect超時(shí)比起修改系統(tǒng)參數(shù)的那種方 法的有點(diǎn)就在于它只用于你的程序之中而不影響系統(tǒng)。