• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            面對(duì)現(xiàn)實(shí),超越自己
            逆水行舟,不進(jìn)則退
            posts - 269,comments - 32,trackbacks - 0

                   采用TCP連接的C/S模式軟件,連接的雙方在連接空閑狀態(tài)時(shí),如果任意一方意外崩潰、當(dāng)機(jī)、網(wǎng)線斷開或路由器故障,另一方無法得知TCP連接已經(jīng)失效,除非繼續(xù)在此連接上發(fā)送數(shù)據(jù)導(dǎo)致錯(cuò)誤返回。很多時(shí)候,這不是我們需要的。我們希望服務(wù)器端和客戶端都能及時(shí)有效地檢測到連接失效,然后優(yōu)雅地完成一些清理工作并把錯(cuò)誤報(bào)告給用戶。
                  如何及時(shí)有效地檢測到一方的非正常斷開,一直有兩種技術(shù)可以運(yùn)用。一種是由TCP協(xié)議層實(shí)現(xiàn)的Keepalive,另一種是由應(yīng)用層自己實(shí)現(xiàn)的心跳包
                  TCP默認(rèn)并不開啟Keepalive功能,因?yàn)殚_啟Keepalive功能需要消耗額外的寬帶和流量,盡管這微不足道,但在按流量計(jì)費(fèi)的環(huán)境下增加了費(fèi)用,另一方面,Keepalive設(shè)置不合理時(shí)可能會(huì)因?yàn)槎虝旱木W(wǎng)絡(luò)波動(dòng)而斷開健康的TCP連接。并且,默認(rèn)的Keepalive超時(shí)需要7,200,000 milliseconds,即2小時(shí),探測次數(shù)為5次。
                  對(duì)于Win2K/XP/2003,可以從下面的注冊(cè)表項(xiàng)找到影響整個(gè)系統(tǒng)所有連接的keepalive參數(shù):
                   [HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Services/Tcpip/Parameters]
                    "KeepAliveTime”=dword:006ddd00
                    "KeepAliveInterval"=dword:000003e8 
                    "MaxDataRetries"="5"

                  對(duì)于實(shí)用的程序來說,2小時(shí)的空閑時(shí)間太長。因此,我們需要手工開啟Keepalive功能并設(shè)置合理的Keepalive參數(shù)。

            // 開啟KeepAlive
            BOOL bKeepAlive = TRUE;
            int nRet = ::setsockopt(socket_handle, SOL_SOCKET, SO_KEEPALIVE, (char*)&bKeepAlive, sizeof(bKeepAlive));

            if (nRet == SOCKET_ERROR)
            {
            return FALSE;
            }


            // 設(shè)置KeepAlive參數(shù)
            tcp_keepalive alive_in = {0};
            tcp_keepalive alive_out 
            = {0};
            alive_in.keepalivetime 
            = 5000// 開始首次KeepAlive探測前的TCP空閉時(shí)間

            alive_in.keepaliveinterval 
            = 1000// 兩次KeepAlive探測間的時(shí)間間隔

            alive_in.onoff 
            = TRUE;
            unsigned 
            long ulBytesReturn = 0;

            nRet 
            = WSAIoctl(socket_handle, SIO_KEEPALIVE_VALS, &alive_in, sizeof(alive_in),
            &alive_out, sizeof(alive_out), &ulBytesReturn, NULL, NULL);
            if (nRet == SOCKET_ERROR)
            {
            return FALSE;


            }



            開啟Keepalive選項(xiàng)之后,對(duì)于使用IOCP模型的服務(wù)器端程序來說,一旦檢測到連接斷開,GetQueuedCompletionStatus函數(shù)將立即返回FALSE,使得服務(wù)器端能及時(shí)清除該連接、釋放該連接相關(guān)的資源。對(duì)于使用select模型的客戶端來說,連接斷開被探測到時(shí),以recv目的阻塞在socket上的select方法將立即返回SOCKET_ERROR,從而得知連接已失效,客戶端程序便有機(jī)會(huì)及時(shí)執(zhí)行清除工作、提醒用戶或重新連接。

             

            另一種技術(shù),由應(yīng)用程序自己發(fā)送心跳包來檢測連接的健康性。客戶端可以在一個(gè)Timer中或低級(jí)別的線程中定時(shí)向發(fā)服務(wù)器發(fā)送一個(gè)短小精悍的包,并等待服務(wù)器的回應(yīng)。客戶端程序在一定時(shí)間內(nèi)沒有收到服務(wù)器回應(yīng)即認(rèn)為連接不可用,同樣,服務(wù)器在一定時(shí)間內(nèi)沒有收到客戶端的心跳包則認(rèn)為客戶端已經(jīng)掉線。

             

            ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

             windows下此處的”非正常斷開”指TCP連接不是以優(yōu)雅的方式斷開,如網(wǎng)線故障等物理鏈路的原因,還有突然主機(jī)斷電等原因.

            有兩種方法可以檢測:

            1.TCP連接雙方定時(shí)發(fā)握手消息

             


            2.利用TCP協(xié)議棧中的KeepAlive探測
            第二種方法簡單可靠,只需對(duì)TCP連接兩個(gè)Socket設(shè)定KeepAlive探測,
            所以本文只講第二種方法在Linux,Window2000下的實(shí)現(xiàn)(在其它的平臺(tái)上沒有作進(jìn)一步的測試)


            Windows 2000平臺(tái)下 頭文件
            #include 
            <mstcpip.h>
            //定義結(jié)構(gòu)及宏
            /*

            struct TCP_KEEPALIVE {
            u_longonoff;
            u_longkeepalivetime;
            u_longkeepaliveinterval;
            }
             ;
            */

                tcp_keepalive live,liveout;  
                live.keepaliveinterval=5000; //每5秒發(fā)一次探測報(bào)文,發(fā)5次沒有回應(yīng),就斷開
                live.keepalivetime=30000;//超過30s沒有數(shù)據(jù),就發(fā)送控測包
                
                live.onoff=TRUE;  
                int Opt = 1;
                int iRet = setsockopt(Accept,SOL_SOCKET,SO_KEEPALIVE,(char *)&Opt,sizeof(int));  
                if(iRet == 0)
                {
                     DWORD dw;
                     if(::WSAIoctl(Accept,SIO_KEEPALIVE_VALS,
                      &live,sizeof(live),&liveout,sizeof(liveout),
                      &dw,NULL,NULL)== SOCKET_ERROR){
                 }  
            }


            ACE下代碼 //by rainfish blog.csdn.net/bat603

             

            int Opt = 1;
            //在測試過程中,發(fā)現(xiàn)檢測的次數(shù)是5次,即下面的設(shè)置中,從最近一次消息開始計(jì)算的10秒后,每次間隔5秒,連續(xù)發(fā)送5次,即35秒發(fā)現(xiàn)網(wǎng)絡(luò)斷了
            tcp_keepalive live,liveout; 
            live.keepaliveinterval=5000//每次檢測的間隔 (單位毫秒)
            live.keepalivetime=10000//第一次開始發(fā)送的時(shí)間(單位毫秒)
            live.onoff=TRUE; 
            int iRet = stream.set_option(SOL_SOCKET,SO_KEEPALIVE,&Opt,sizeof(int)); 
            if(iRet == 0)

                  DWORD dw;
                  //此處顯示了在ACE下獲取套接字的方法,即句柄的(SOCKET)化就是句柄
                  if(WSAIoctl((SOCKET)h,SIO_KEEPALIVE_VALS,&live,sizeof(live),
                  &liveout,sizeof(liveout),&dw,NULL,NULL)== SOCKET_ERROR)
                  {
                        //Delete Client 
                        return
                  
            }

             


            Linux平臺(tái)下

            #include "/usr/include/linux/tcp.h"
            #include "/usr/include/linux/socket.h"
            ////KeepAlive實(shí)現(xiàn),單位秒
            //下面代碼要求有ACE,如果沒有包含ACE,則請(qǐng)把用到的ACE函數(shù)改成linux相應(yīng)的接口
            int keepAlive = 1;//設(shè)定KeepAlive
            int keepIdle = 5;//開始首次KeepAlive探測前的TCP空閉時(shí)間
            int keepInterval = 5;//兩次KeepAlive探測間的時(shí)間間隔
            int keepCount = 3;//判定斷開前的KeepAlive探測次數(shù)
            if(setsockopt(s,SOL_SOCKET,SO_KEEPALIVE,(void*)&keepAlive,sizeof(keepAlive)) == -1)
            {
                  ACE_DEBUG ((LM_INFO,
                  ACE_TEXT ("(%P|%t) setsockopt SO_KEEPALIVE error!/n")));
            }
            if(setsockopt(s,SOL_TCP,TCP_KEEPIDLE,(void *)&keepIdle,sizeof(keepIdle)) == -1)
            {
                  ACE_DEBUG ((LM_INFO,
                  ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPIDLE error!/n")));
            }
            if(setsockopt(s,SOL_TCP,TCP_KEEPINTVL,(void *)&keepInterval,sizeof(keepInterval)) == -1)
            {
                  ACE_DEBUG ((LM_INFO,
                  ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPINTVL error!/n")));
            }
            if(setsockopt(s,SOL_TCP,TCP_KEEPCNT,(void *)&keepCount,sizeof(keepCount)) == -1)
            {
                  ACE_DEBUG ((LM_INFO,
                  ACE_TEXT ("(%P|%t)setsockopt TCP_KEEPCNT error!/n")));
            }

             

            ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++


            本文轉(zhuǎn)自:http://www.shnenglu.com/API/archive/2013/08/13/202516.html

             

            posted on 2013-08-14 19:14 王海光 閱讀(1222) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++
            久久最新精品国产| 亚洲国产另类久久久精品黑人| 亚洲精品无码久久久久久| 亚洲精品97久久中文字幕无码| 无码久久精品国产亚洲Av影片 | 婷婷久久久亚洲欧洲日产国码AV| 久久久久久久91精品免费观看| 久久久久人妻一区精品色| 三级韩国一区久久二区综合| 奇米影视7777久久精品| 免费一级做a爰片久久毛片潮| 国产精品久久久久久福利漫画| 大香伊人久久精品一区二区| 四虎国产精品免费久久5151| 久久久久国产| 精品久久久久久无码人妻蜜桃| 99久久无码一区人妻a黑| 国内精品久久久久影院薰衣草| 久久亚洲国产精品123区| 色婷婷综合久久久久中文一区二区| 狠狠色噜噜狠狠狠狠狠色综合久久| 欧美午夜精品久久久久免费视 | 亚洲人成网站999久久久综合| 伊人久久大香线蕉av一区| 国产精自产拍久久久久久蜜| 国产精品视频久久久| 色8激情欧美成人久久综合电| 91久久精品91久久性色| 久久亚洲AV无码精品色午夜| 伊人久久大香线蕉影院95| 91久久精品国产成人久久| 人妻无码久久一区二区三区免费| 91麻豆精品国产91久久久久久| 亚洲中文字幕无码久久精品1| 婷婷久久综合| 久久99精品免费一区二区 | 久久精品国产亚洲精品| 国产欧美久久一区二区| 久久99国内精品自在现线| 色综合久久久久久久久五月| 亚洲国产精品无码成人片久久|