Socket編程實踐(9) --套接字IO超時設置方法

引:超時設置3種方案

1. alarm超時設置方法

  1. //代碼實現: 這種方式較少用  
  2. void sigHandlerForSigAlrm(int signo)  
  3. {  
  4.     return ;  
  5. }  
  6.   
  7. signal(SIGALRM, sigHandlerForSigAlrm);  
  8. alarm(5);  
  9. int ret = read(sockfd, buf, sizeof(buf));  
  10. if (ret == -1 && errno == EINTR)  
  11. {  
  12.     // 超時被時鐘打斷  
  13.     errno = ETIMEDOUT;  
  14. }  
  15. else if (ret >= 0)  
  16. {  
  17.     // 正常返回(沒有超時), 則將鬧鐘關閉  
  18.     alarm(0);  
  19. }  

2. 套接字選項: SO_SNDTIMEO, SO_RCVTIMEO

調用setsockopt設置讀/寫超時時間

  1. //示例: read超時  
  2. int seconds = 5;  
  3. if (setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &seconds, sizeof(seconds)) == -1)  
  4.     err_exit("setsockopt error");  
  5. int ret = read(sockfd, buf, sizeof(buf));  
  6. if (ret == -1 && errno == EWOULDBLOCK)  
  7. {  
  8.     // 超時,被時鐘打斷  
  9.     errno = ETIMEDOUT;  
  10. }  

3. select方式[重點]

 

_timeout函數封裝

1. read_timeout封裝

  1. /** 
  2.  *read_timeout - 讀超時檢測函數, 不包含讀操作 
  3.  *@fd: 文件描述符 
  4.  *@waitSec: 等待超時秒數, 0表示不檢測超時 
  5.  *成功(未超時)返回0, 失敗返回-1, 超時返回-1 並且 errno = ETIMEDOUT 
  6. **/  
  7. int read_timeout(int fd, long waitSec)  
  8. {  
  9.     int returnValue = 0;  
  10.     if (waitSec > 0)  
  11.     {  
  12.         fd_set readSet;  
  13.         FD_ZERO(&readSet);  
  14.         FD_SET(fd,&readSet);    //添加  
  15.   
  16.         struct timeval waitTime;  
  17.         waitTime.tv_sec = waitSec;  
  18.         waitTime.tv_usec = 0;       //將微秒設置爲0(不進行設置),如果設置了,時間會更加精確  
  19.         do  
  20.         {  
  21.             returnValue = select(fd+1,&readSet,NULL,NULL,&waitTime);  
  22.         }  
  23.         while(returnValue < 0 && errno == EINTR);   //等待被(信號)打斷的情況, 重啓select  
  24.   
  25.         if (returnValue == 0)   //在waitTime時間段中一個事件也沒到達  
  26.         {  
  27.             returnValue = -1;   //返回-1  
  28.             errno = ETIMEDOUT;  
  29.         }  
  30.         else if (returnValue == 1)  //在waitTime時間段中有事件產生  
  31.             returnValue = 0;    //返回0,表示成功  
  32.         // 如果(returnValue == -1) 並且 (errno != EINTR), 則直接返回-1(returnValue)  
  33.     }  
  34.   
  35.     return returnValue;  
  36. }  

2. write_timeout封裝

  1. /** 
  2.  *write_timeout - 寫超時檢測函數, 不包含寫操作 
  3.  *@fd: 文件描述符 
  4.  *@waitSec: 等待超時秒數, 0表示不檢測超時 
  5.  *成功(未超時)返回0, 失敗返回-1, 超時返回-1 並且 errno = ETIMEDOUT 
  6. **/  
  7. int write_timeout(int fd, long waitSec)  
  8. {  
  9.     int returnValue = 0;  
  10.     if (waitSec > 0)  
  11.     {  
  12.         fd_set writeSet;  
  13.         FD_ZERO(&writeSet);      //清零  
  14.         FD_SET(fd,&writeSet);    //添加  
  15.   
  16.         struct timeval waitTime;  
  17.         waitTime.tv_sec = waitSec;  
  18.         waitTime.tv_usec = 0;  
  19.         do  
  20.         {  
  21.             returnValue = select(fd+1,NULL,&writeSet,NULL,&waitTime);  
  22.         } while(returnValue < 0 && errno == EINTR); //等待被(信號)打斷的情況  
  23.   
  24.         if (returnValue == 0)   //在waitTime時間段中一個事件也沒到達  
  25.         {  
  26.             returnValue = -1;   //返回-1  
  27.             errno = ETIMEDOUT;  
  28.         }  
  29.         else if (returnValue == 1)  //在waitTime時間段中有事件產生  
  30.             returnValue = 0;    //返回0,表示成功  
  31.     }  
  32.   
  33.     return returnValue;  
  34. }  

3. accept_timeout函數封裝

  1. /** 
  2.  *accept_timeout - 帶超時的accept 
  3.  *@fd: 文件描述符 
  4.  *@addr: 輸出參數, 返回對方地址 
  5.  *@waitSec: 等待超時秒數, 0表示不使用超時檢測, 使用正常模式的accept 
  6.  *成功(未超時)返回0, 失敗返回-1, 超時返回-1 並且 errno = ETIMEDOUT 
  7. **/  
  8. int accept_timeout(int fd, struct sockaddr_in *addr, long waitSec)  
  9. {  
  10.     int returnValue = 0;  
  11.     if (waitSec > 0)  
  12.     {  
  13.         fd_set acceptSet;  
  14.         FD_ZERO(&acceptSet);  
  15.         FD_SET(fd,&acceptSet);    //添加  
  16.   
  17.         struct timeval waitTime;  
  18.         waitTime.tv_sec = waitSec;  
  19.         waitTime.tv_usec = 0;  
  20.         do  
  21.         {  
  22.             returnValue = select(fd+1,&acceptSet,NULL,NULL,&waitTime);  
  23.         }  
  24.         while(returnValue < 0 && errno == EINTR);  
  25.   
  26.         if (returnValue == 0)  //在waitTime時間段中沒有事件產生  
  27.         {  
  28.             errno = ETIMEDOUT;  
  29.             return -1;  
  30.         }  
  31.         else if (returnValue == -1) // error  
  32.             return -1;  
  33.     }  
  34.   
  35.     /**select正確返回: 
  36.         表示有select所等待的事件發生:對等方完成了三次握手, 
  37.         客戶端有新的鏈接建立,此時再調用accept就不會阻塞了 
  38.     */  
  39.     socklen_t socklen = sizeof(struct sockaddr_in);  
  40.     if (addr != NULL)  
  41.         returnValue = accept(fd,(struct sockaddr *)addr,&socklen);  
  42.     else  
  43.         returnValue = accept(fd,NULL,NULL);  
  44.   
  45.     return returnValue;  
  46. }  

4. connect_timeout函數封裝

(1)我們爲什麼需要這個函數?

   TCP/IP在客戶端連接服務器時,如果發生異常,connect(如果是在默認阻塞的情況下)返回的時間是RTT(相當於客戶端阻塞了這麼長的時間,客戶需要等待這麼長的時間,顯然這樣的客戶端用戶體驗並不好(完成三次握手需要使用1.5RTT時間));會造成嚴重的軟件質量下降.

(2)怎樣實現connect_timeout?

   1)sockfd首先變成非阻塞的; 然後試着進行connect,如果網絡狀況良好,則立刻建立鏈接並返回,如果網絡狀況不好,則鏈接不會馬上建立,這時需要我們的參與:調用select,設置等待時間,通過select管理者去監控sockfd,一旦能夠建立鏈接,則馬上返回,然後建立鏈接,這樣就會大大提高我們的軟件質量.

   2)需要注意:select機制監控到sockfd可寫(也就是可以建立鏈接時),並不代表調用connect就一定能夠成功(造成sockfd可寫有兩種情況: a.真正的鏈接可以建立起來了; b.建立鏈接的過程中發生錯誤,然後錯誤會回寫錯誤信息,造成sockfd可寫);

   通過調用getsockopt做一個容錯即可(見下例)!

(3)代碼實現:

  1. /**設置文件描述符fd爲非阻塞/阻塞模式**/  
  2. bool setUnBlock(int fd, bool unBlock)  
  3. {  
  4.     int flags = fcntl(fd,F_GETFL);  
  5.     if (flags == -1)  
  6.         return false;  
  7.   
  8.     if (unBlock)  
  9.         flags |= O_NONBLOCK;  
  10.     else  
  11.         flags &= ~O_NONBLOCK;  
  12.   
  13.     if (fcntl(fd,F_SETFL,flags) == -1)  
  14.         return false;  
  15.     return true;  
  16. }  
  17. /** 
  18.  *connect_timeout - connect 
  19.  *@fd: 文件描述符 
  20.  *@addr: 要連接的對方地址 
  21.  *@waitSec: 等待超時秒數, 0表示使用正常模式的accept 
  22.  *成功(未超時)返回0, 失敗返回-1, 超時返回-1 並且 errno = ETIMEDOUT 
  23. **/  
  24. int connect_timeout(int fd, struct sockaddr_in *addr, long waitSec)  
  25. {  
  26.     if (waitSec > 0)    //設置爲非阻塞模式  
  27.         setUnBlock(fd, true);  
  28.   
  29.     socklen_t addrLen = sizeof(struct sockaddr_in);  
  30.     //首先嚐試着進行鏈接  
  31.     int returnValue = connect(fd,(struct sockaddr *)addr,addrLen);  
  32.     //如果首次嘗試失敗(並且errno == EINPROGRESS表示連接正在處理當中),則需要我們的介入  
  33.     if (returnValue < 0 && errno == EINPROGRESS)  
  34.     {  
  35.         fd_set connectSet;  
  36.         FD_ZERO(&connectSet);  
  37.         FD_SET(fd,&connectSet);  
  38.         struct timeval waitTime;  
  39.         waitTime.tv_sec = waitSec;  
  40.         waitTime.tv_usec = 0;  
  41.         do  
  42.         {  
  43.             /*一旦建立鏈接,則套接字可寫*/  
  44.             returnValue = select(fd+1, NULL, &connectSet, NULL, &waitTime);  
  45.         }  
  46.         while (returnValue < 0 && errno == EINTR);  
  47.         if (returnValue == -1) //error  
  48.             return -1;  
  49.         else if (returnValue == 0)   //超時  
  50.         {  
  51.             returnValue = -1;  
  52.             errno = ETIMEDOUT;  
  53.         }  
  54.         else if (returnValue == 1)  //正確返回,有一個套接字可寫  
  55.         {  
  56.             /**由於connectSet只有一個文件描述符, 因此FD_ISSET的測試也就省了**/  
  57.   
  58.             /**注意:套接字可寫有兩種情況: 
  59.                 1.連接建立成功 
  60.                 2.套接字產生錯誤(但是此時select是正確的, 因此錯誤信息沒有保存在errno中),需要調用getsockopt獲取 
  61.             */  
  62.             int err;  
  63.             socklen_t errLen = sizeof(err);  
  64.             int sockoptret = getsockopt(fd,SOL_SOCKET,SO_ERROR,&err,&errLen);  
  65.             if (sockoptret == -1)  
  66.                 return -1;  
  67.   
  68.             // 測試err的值  
  69.             if (err == 0)   //確實是鏈接建立成功  
  70.                 returnValue = 0;  
  71.             else    //連接產生了錯誤  
  72.             {  
  73.                 errno = err;  
  74.                 returnValue = -1;  
  75.             }  
  76.         }  
  77.     }  
  78.     if (waitSec > 0)  
  79.         setUnBlock(fd, false);  
  80.     return returnValue;  
  81. }  

  1. /**測試:使用connect_timeout的client端完整代碼(server端如前)**/  
  2. int main()  
  3. {  
  4.     int sockfd = socket(AF_INET, SOCK_STREAM, 0);  
  5.     if (sockfd == -1)  
  6.         err_exit("socket error");  
  7.   
  8.     struct sockaddr_in serverAddr;  
  9.     serverAddr.sin_family = AF_INET;  
  10.     serverAddr.sin_port = htons(8001);  
  11.     serverAddr.sin_addr.s_addr = inet_addr("127.0.0.1");  
  12.     int ret = connect_timeout(sockfd, &serverAddr, 5);  
  13.     if (ret == -1 && errno == ETIMEDOUT)  
  14.     {  
  15.         cerr << "timeout..." << endl;  
  16.         err_exit("connect_timeout error");  
  17.     }  
  18.     else if (ret == -1)  
  19.         err_exit("connect_timeout error");  
  20.   
  21.     //獲取並打印對端信息  
  22.     struct sockaddr_in peerAddr;  
  23.     socklen_t peerLen = sizeof(peerAddr);  
  24.     if (getpeername(sockfd, (struct sockaddr *)&peerAddr, &peerLen) == -1)  
  25.         err_exit("getpeername");  
  26.     cout << "Server information: " << inet_ntoa(peerAddr.sin_addr)  
  27.                  << ", " << ntohs(peerAddr.sin_port) << endl;  
  28.     close(sockfd);  
  29. }  

附-RTT(Round-Trip Time)介紹:

   RTT往返時延:在計算機網絡中它是一個重要的性能指標,表示從發送端發送數據開始,到發送端收到來自接收端的確認(接收端收到數據後便立即發送確認),總共經歷的時延。

   RTT由三個部分決定:即鏈路的傳播時間、末端系統的處理時間以及路由器的緩存中的排隊和處理時間。其中,前面兩個部分的值作爲一個TCP連接相對固定,路由器的緩存中的排隊和處理時間會隨着整個網絡擁塞程度的變化而變化。所以RTT的變化在一定程度上反映了網絡擁塞程度的變化。簡單來說就是發送方從發送數據開始,到收到來自接受方的確認信息所經歷的時間。

發佈了52 篇原創文章 · 獲贊 10 · 訪問量 9萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章