etsockopt :SO_LINGER 選項設置

轉載:https://blog.csdn.net/factor2000/article/details/3929816

setsockopt 設置 SO_LINGER 選項

 

   此選項指定函數close對面向連接的協議如何操作(如TCP)。內核缺省close操作是立即返回,如果有數據殘留在套接口緩衝區中則系統將試着將這些數據發送給對方。

 

SO_LINGER選項用來改變此缺省設置。使用如下結構:

struct linger {

     int l_onoff; /* 0 = off, nozero = on */

     int l_linger; /* linger time */

};

 

有下列三種情況:

1、設置 l_onoff爲0,則該選項關閉,l_linger的值被忽略,等於內核缺省情況,close調用會立即返回給調用者,如果可能將會傳輸任何未發送的數據;

 

2、設置 l_onoff爲非0,l_linger爲0,則套接口關閉時TCP夭折連接,TCP將丟棄保留在套接口發送緩衝區中的任何數據併發送一個RST給對方,而不是通常的四分組終止序列,這避免了TIME_WAIT狀態;

 

3、設置 l_onoff 爲非0,l_linger爲非0,當套接口關閉時內核將拖延一段時間(由l_linger決定)。如果套接口緩衝區中仍殘留數據,進程將處於睡眠狀態,直 到(a)所有數據發送完且被對方確認,之後進行正常的終止序列(描述字訪問計數爲0)或(b)延遲時間到。此種情況下,應用程序檢查close的返回值是非常重要的,如果在數據發送完並被確認前時間到,close將返回EWOULDBLOCK錯誤且套接口發送緩衝區中的任何數據都丟失。close的成功返回僅告訴我們發送的數據(和FIN)已由對方TCP確認,它並不能告訴我們對方應用進程是否已讀了數據。如果套接口設爲非阻塞的,它將不等待close完成。

 

註釋:l_linger的單位依賴於實現: 4.4BSD假設其單位是時鐘滴答(百分之一秒),但Posix.1g規定單位爲秒。

 

下面的代碼是一個使用SO_LINGER選項的例子,使用30秒的超時時限:
#define TRUE     1
#define FALSE    0
int z; /* Status code
*/ int s;       /* Socket s */
struct linger so_linger;
...
so_linger.l_onoff = TRUE;
so_linger.l_linger = 30;
z = setsockopt(s,
    SOL_SOCKET,
    SO_LINGER,
    &so_linger,
    sizeof so_linger);
if ( z )
   perror("setsockopt(2)");

下面的例子顯示瞭如何設置SO_LINGER的值來中止套接口s上的當前連接:
#define TRUE     1
#define FALSE    0
int z; /* Status code */
int s;       /* Socket s */
struct linger so_linger;
...
so_linger.l_onoff = TRUE;
so_linger.l_linger = 0;
z = setsockopt(s,
    SOL_SOCKET,
    SO_LINGER,
    &so_linger,
    sizeof so_linger);
if ( z )
    perror("setsockopt(2)");
    close(s); /* Abort connection */

在上面的這個例子中,當調用close函數時,套接口s會立即中止。中止的語義是通過將超時值設置爲0來實現的。

 

 

 

 

/********** WINDOWS **********/

 

 

/* 當連接中斷時,需要延遲關閉(linger)以保證所有數據都被傳輸,所以需要打開SO_LINGER這個選項;  
 * //注:大致意思就是說SO_LINGER選項用來設置當調用closesocket時是否馬上關閉socket; 
 * linger的結構在/usr/include/linux/socket.h中定義://注:這個結構就是SetSocketOpt中的Data的數據結構 
 *  struct linger 
 *  { 
 *   int l_onoff;  /* Linger active */       //低字節,0和非0,用來表示是否延時關閉socket
 *   int l_linger; /* How long to linger */   //高字節,延時的時間數,單位爲秒
 *  }; 
 *  如果l_onoff爲0,則延遲關閉特性就被取消。

 *   如果非零,則允許套接口延遲關閉; l_linger字段則指明延遲關閉的時間 
 */


更具體的描述如下:
1、若設置了SO_LINGER(亦即linger結構中的l_onoff域設爲非零),並設置了零超時間隔,則closesocket()不被阻塞立即執行,不論是否有排隊數據未發送或未被確認。這種關閉方式稱爲“強制”或“失效”關閉,因爲套接口的虛電路立即被複位,且丟失了未發送的數據。在遠端的recv()調用將以WSAECONNRESET出錯。

2、若設置了SO_LINGER並確定了非零的超時間隔,則closesocket()調用阻塞進程,直到所剩數據發送完畢或超時。這種關閉稱爲“優雅”或“從容”關閉。請注意如果套接口置爲非阻塞且SO_LINGER設爲非零超時,則closesocket()調用將以WSAEWOULDBLOCK錯誤返回。

3、若在一個流類套接口上設置了SO_DONTLINGER(也就是說將linger結構的l_onoff域設爲零),則closesocket()調用立即返回。但是,如果可能,排隊的數據將在套接口關閉前發送。請注意,在這種情況下WINDOWS套接口實現將在一段不確定的時間內保留套接口以及其他資源,這對於想用所以套接口的應用程序來說有一定影響。
 

SO_DONTLINGER 若爲真,則SO_LINGER選項被禁止。
SO_LINGER延遲關閉連接 struct linger上面這兩個選項影響close行爲;


     選項          間隔    關閉方式  等待關閉與否
  SO_DONTLINGER   不關心     優雅         否
  SO_LINGER        零        強制         否
  SO_LINGER       非零       優雅         是
————————————————
版權聲明:本文爲CSDN博主「factor2000」的原創文章,遵循CC 4.0 BY-SA版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/factor2000/article/details/3929816

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章