• <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>

            Bugs

            MMORPG game develop.

            網(wǎng)絡(luò)編程 心得2

            如果客戶端必須使用綁定端口,
            那么在關(guān)閉的時(shí)候,會(huì)經(jīng)歷TIME_WAIT的過(guò)程,一般windows下是2分鐘,這段時(shí)間,客戶端connect的時(shí)候,會(huì)出錯(cuò)(WSAEADDRINUSE:10048),
            怎么不經(jīng)歷這個(gè)狀態(tài)呢?
            使用下面代碼:

              // 如果要已經(jīng)處于連接狀態(tài)的soket在調(diào)用closesocket后強(qiáng)制關(guān)閉,不經(jīng)歷TIME_WAIT的過(guò)程:
              BOOL bDontLinger = FALSE;
              if (setsockopt(s,SOL_SOCKET,SO_DONTLINGER,(const char*)&bDontLinger,sizeof(BOOL))< 0) {
               wsaperror("setsockopt");
               break;
              }

            posted on 2008-04-24 18:33 Bugs 閱讀(4536) 評(píng)論(7)  編輯 收藏 引用

            評(píng)論

            # re: 網(wǎng)絡(luò)編程 心得2[未登錄](méi) 2008-04-25 08:57 christanxw

            SO_DONTLINGER的設(shè)置將會(huì)使得未發(fā)送的系統(tǒng)緩沖區(qū)數(shù)據(jù)丟失。更好的還是設(shè)置為重用端口。  回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2[未登錄](méi) 2008-04-25 10:47 Bugs

            @christanxw 重用端口,設(shè)置過(guò)了,但在windows(只做了win測(cè)試)平臺(tái)下,是沒(méi)有作用的。

            // 下面是測(cè)試代碼,在window平臺(tái)下測(cè)試過(guò)了。
            // 沒(méi)有l(wèi)inux環(huán)境,有環(huán)境的朋友可以試試。

            #ifdef WIN32
            #ifndef WIN32_LEAN_AND_MEAN
            #define WIN32_LEAN_AND_MEAN
            #endif

            #pragma comment(lib, "WS2_32.lib")
            #pragma warning( disable : 4996 )

            #endif

            #include <sys/types.h>
            #include <stdlib.h>
            #include <stdio.h>
            #include <string.h>
            #ifdef WIN32
            #include <winsock2.h>
            #else
            #include <sys/socket.h>
            #include <netinet/in.h>
            #endif

            #ifndef WIN32
            #define wsaperror perror
            #else
            void wsaperror(char *);
            #endif

            int main(int argc, char *argv[])
            {
            #ifdef WIN32
            WSADATA wsaData;
            #endif
            int succeeded = 0;
            struct sockaddr_in addr = {0};
            int i;

            // 遠(yuǎn)端地址 隨便鏈接一個(gè)處于listen狀態(tài)的
            memset(&addr, 0, sizeof(addr));
            addr.sin_family = AF_INET;
            addr.sin_addr.s_addr = inet_addr("192.168.170.13");
            addr.sin_port = htons(135);

            #ifdef WIN32
            if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {
            wsaperror("WSAStartup");
            exit(1);
            }
            #endif
            while (1) {
            int s = socket(AF_INET, SOCK_STREAM, 0);
            struct sockaddr_in local_addr = {0};


            #ifdef WIN32
            int name_len = sizeof(local_addr);
            #else
            socklen_t name_len = sizeof(local_addr);
            #endif

            if (s < 0) {
            wsaperror("socket");
            break;
            }

            // 下面的代碼,有l(wèi)inux環(huán)境的朋友,注釋掉,不設(shè)置DontLinger試試。
            //
            // 如果要已經(jīng)處于連接狀態(tài)的soket在調(diào)用closesocket后強(qiáng)制關(guān)閉,不經(jīng)歷TIME_WAIT的過(guò)程:
            BOOL bDontLinger = FALSE;
            if (setsockopt(s,SOL_SOCKET,SO_DONTLINGER,(const char*)&bDontLinger,sizeof(BOOL))< 0) {
            wsaperror("setsockopt");
            break;
            }

            i = 1;
            if (setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (char *) &i, sizeof(i)) < 0) {
            wsaperror("setsockopt");
            break;
            }


            // 本地地址
            local_addr.sin_family = AF_INET;
            local_addr.sin_addr.s_addr = inet_addr("192.168.170.13");
            local_addr.sin_port = htons(10000);

            if (bind(s, (struct sockaddr *) &local_addr, sizeof(local_addr)) < 0) {
            wsaperror("bind");
            break;
            }

            if (getsockname(s, (struct sockaddr *) &local_addr, &name_len) < 0) {
            break;
            }

            if (connect(s, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
            //fprintf(stderr, "Local port failed: %s : %d\n", inet_ntoa(local_addr.sin_addr), ntohs(local_addr.sin_port));
            wsaperror("connect");
            break;
            }

            succeeded++;

            /* shutdown(s, 2); */
            #ifdef WIN32
            closesocket(s);
            #else
            close(s);
            #endif

            fprintf(stderr, "%d\n", ntohs(local_addr.sin_port));
            fflush(stdout);
            }

            fprintf(stderr, "%d connections succeeded\n", succeeded);

            #ifdef WIN32
            WSACleanup();
            #endif
            exit(1);
            }

            #ifdef WIN32
            void wsaperror(char *str)
            {
            int err = WSAGetLastError();
            char *errstr;

            switch (err) {
            case WSAEADDRINUSE:
            errstr = "WSAEADDRINUSE";
            break;
            case WSAENOBUFS:
            errstr = "WSAENOBUFS";
            break;
            default:
            errstr = "Unknown error";
            break;
            }

            fprintf(stderr, "%s: Error %d (%s)\n", str, err, errstr);
            }
            #endif
              回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2 2008-04-25 10:59 bugs_killer

            這個(gè)方法太猥瑣了...

            SO_DONTLINGER 這個(gè)選項(xiàng)會(huì)導(dǎo)致 數(shù)據(jù)的接收端會(huì)丟失數(shù)據(jù).  回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2 2008-04-25 11:03 Bugs

            這里在說(shuō)說(shuō) REUSERADDR,
            ACE網(wǎng)絡(luò)編程里說(shuō)過(guò),windows 在 2000以后, REUSERADDR 標(biāo)識(shí)默認(rèn)為1的。

            但我在2003下測(cè)試的時(shí)候:
            1.如果客戶端使用了綁定端口,沒(méi)有設(shè)置REUSERADDR為1時(shí),bind的時(shí)候,會(huì)返回失敗,反之成功。
            但成功失敗都不會(huì)影響客戶端調(diào)用connect,只是成功后,則使用你指定的ADDR進(jìn)行connect,失敗則是隨機(jī)分配的PORT進(jìn)行連接。

            2.但設(shè)置了REUSERADDR的作用也僅僅是上面的功能。
            當(dāng)在客戶端主動(dòng)斷開(kāi)連接后,處于TIME_WAIT狀態(tài)時(shí),在2MSL時(shí)間里,你綁定(bind)PORT的時(shí)候,雖然不會(huì)失敗,但在connect的時(shí)候,會(huì)返回WSAEADDRINUSE(10048)錯(cuò)誤。

            10048的描述是:通常每個(gè)套接字地址(協(xié)議/網(wǎng)絡(luò)地址/端口)只允許使用一次。

            這里就跟《網(wǎng)絡(luò)編程 心得1》里說(shuō)的,處于主動(dòng)關(guān)閉后的Time_wait狀態(tài)下,
            如果設(shè)置了REUSERADDR為1時(shí),是可以復(fù)用prot進(jìn)行連接的。
            網(wǎng)上找了很多資料,說(shuō)windows平臺(tái)下這里是一個(gè)bug。
            在linux平臺(tái)下,還沒(méi)有環(huán)境做測(cè)試,有時(shí)間找個(gè)環(huán)境測(cè)試下,應(yīng)該是可以復(fù)用PORT,進(jìn)行連接的。

            那么現(xiàn)在windows平臺(tái)下怎么避免2MSL時(shí)間內(nèi)不能連接的問(wèn)題呢。
            BOOL bDontLinger = FALSE;
            setsockopt(s,SOL_SOCKET,SO_DONTLINGER,(const char*)&bDontLinger,sizeof(BOOL));
            這樣做,使主動(dòng)關(guān)閉方在close后,不經(jīng)歷TIME_WAIT狀態(tài),也就是說(shuō)這樣可能會(huì)丟失數(shù)據(jù),比如關(guān)閉四次握手最后一次ACK如果在路由中迷途,就會(huì)影響到他的重發(fā)。
            但不這樣,還有其他辦法嗎?  回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2 2008-05-07 09:28 abettor

            @Bugs
            據(jù)我所知,貌似沒(méi)有。
            如果真的那么恨TIME_WAIT,我覺(jué)得可以在應(yīng)用層上配合一下。也就是說(shuō),socket上仍然設(shè)置SO_DONTLINGER這個(gè)選項(xiàng),但是,在制定應(yīng)用層協(xié)議的時(shí)候,把主動(dòng)closesocket的人定為最后一個(gè)接收數(shù)據(jù)包的人,不接收到最后一個(gè)包,就不closesocket。
            但是,話說(shuō)回來(lái),這樣所消耗的資源貌似比TIME_WAIT還大,呵呵。恕我直言,TCP/IP協(xié)議問(wèn)題再多,也是N多大牛設(shè)計(jì)并完善出來(lái)的,不是耍一些小聰明就能將人家“改造”的更優(yōu)秀的。  回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2[未登錄](méi) 2008-05-08 17:15 Bugs

            @abettor
            謝謝你的建議,我又對(duì)這個(gè)部分有了一個(gè)重新的了解了。  回復(fù)  更多評(píng)論   

            # re: 網(wǎng)絡(luò)編程 心得2 2008-05-09 18:20 hellfire

            這樣會(huì)導(dǎo)致緩沖區(qū)中的數(shù)據(jù)發(fā)送不出去的。

            交給應(yīng)用層來(lái)處理關(guān)閉,應(yīng)用層的開(kāi)發(fā)負(fù)擔(dān)太大了。

            按照msdn 上的bug描述,應(yīng)該是由于你在服務(wù)器失敗之后,沒(méi)有關(guān)閉對(duì)應(yīng)客戶端的socket. 才導(dǎo)致的這個(gè)問(wèn)題
              回復(fù)  更多評(píng)論   

            色综合合久久天天给综看| 精品熟女少妇AV免费久久| 亚洲va久久久噜噜噜久久天堂| A级毛片无码久久精品免费| 久久99精品国产麻豆| 久久精品www人人爽人人| 久久久久亚洲AV无码网站| 久久婷婷五月综合色奶水99啪| 久久综合鬼色88久久精品综合自在自线噜噜 | 无码精品久久一区二区三区 | 久久丫精品国产亚洲av| 99久久夜色精品国产网站 | 婷婷久久综合| 99精品久久久久久久婷婷| 亚洲愉拍99热成人精品热久久 | 亚洲AV日韩AV天堂久久| 国产Av激情久久无码天堂| 色综合久久88色综合天天| 久久AAAA片一区二区| 久久婷婷色香五月综合激情| 伊人久久综合成人网| 久久国产精品99精品国产| 国产精品美女久久久久av爽| 人妻丰满?V无码久久不卡| 亚洲AV无码成人网站久久精品大| 久久精品蜜芽亚洲国产AV| 九九久久精品无码专区| 国产精品久久久久蜜芽| 久久福利青草精品资源站免费| 国产伊人久久| 国内精品久久久久久99蜜桃| 久久久精品视频免费观看| 中文字幕无码免费久久| 一本大道加勒比久久综合| 国产精品久久久久免费a∨| 久久精品国产一区| 99精品国产免费久久久久久下载| 久久九九青青国产精品| 久久久午夜精品| 狠狠人妻久久久久久综合蜜桃| 亚洲国产欧美国产综合久久|