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

            Focus on ACE

            訂閱 ace-china
            電子郵件:
            瀏覽存于 groups.google.com 上的所有帖子

            C++博客 首頁 新隨筆 聯(lián)系 聚合 管理
              64 Posts :: 3 Stories :: 22 Comments :: 0 Trackbacks
            socket編程:SO_REUSEADDR例解

            ????????????????????????????????????????????kevintz?2000-6-19

            ????網(wǎng)友vmstat多次提出了這個(gè)問題:SO_REUSEADDR有什么用處和怎么使用。而
            且很多網(wǎng)友在編寫網(wǎng)絡(luò)程序時(shí)也會(huì)遇到這個(gè)問題。所以特意寫了這么一篇文章,
            希望能夠解答一些人的疑難。
            ????其實(shí)這個(gè)問題在Richard?Stevens的《Unix網(wǎng)絡(luò)編程指南》卷一里有很詳細(xì)的
            解答(中文版P166-168頁)。這里我只是寫幾個(gè)基本的例子來驗(yàn)證這個(gè)問題。
            ????首先聲明一個(gè)問題:當(dāng)兩個(gè)socket的address和port相沖突,而你又想重用地
            址和端口,則舊的socket和新的socket都要已經(jīng)被設(shè)置了SO_REUSEADDR特性,只
            有兩者之一有這個(gè)特性還是有問題的。
            ????SO_REUSEADDR可以用在以下四種情況下。
            ????(摘自《Unix網(wǎng)絡(luò)編程》卷一,即UNPv1)
            ????1、當(dāng)有一個(gè)有相同本地地址和端口的socket1處于TIME_WAIT狀態(tài)時(shí),而你啟
            動(dòng)的程序的socket2要占用該地址和端口,你的程序就要用到該選項(xiàng)。
            ????2、SO_REUSEADDR允許同一port上啟動(dòng)同一服務(wù)器的多個(gè)實(shí)例(多個(gè)進(jìn)程)。但
            每個(gè)實(shí)例綁定的IP地址是不能相同的。在有多塊網(wǎng)卡或用IP?Alias技術(shù)的機(jī)器可
            以測試這種情況。
            ????3、SO_REUSEADDR允許單個(gè)進(jìn)程綁定相同的端口到多個(gè)socket上,但每個(gè)soc
            ket綁定的ip地址不同。這和2很相似,區(qū)別請看UNPv1。
            ????4、SO_REUSEADDR允許完全相同的地址和端口的重復(fù)綁定。但這只用于UDP的
            多播,不用于TCP。

            例子1:測試上面第一種情況。
            #include?<netinet/in.h>?
            #include?
            <sys/socket.h>?
            #include?
            <time.h>?
            #include?
            <stdio.h>?
            #include?
            <string.h>?

            #define?MAXLINE?100?

            int?main(int?argc,?char**?argv)?
            {?
            ???
            int?listenfd,connfd;?
            ???
            struct?sockaddr_in?servaddr;?
            ???
            char?buff[MAXLINE+1];?
            ???time_t?ticks;?
            ???unsigned?
            short?port;?
            ???
            int?flag=1,len=sizeof(int);?

            ???port
            =10013;?
            ???
            if(?(listenfd=socket(AF_INET,SOCK_STREAM,0))?==?-1)?
            ???
            {?
            ?????perror(
            "socket");?
            ?????exit(
            1);?
            ???}
            ?
            ???bzero(
            &servaddr,sizeof(servaddr));?
            ???servaddr.sin_family
            =AF_INET;?
            ???servaddr.sin_addr.s_addr
            =htonl(INADDR_ANY);?
            ???servaddr.sin_port
            =htons(port);?
            ???
            if(?setsockopt(listenfd,?SOL_SOCKET,?SO_REUSEADDR,?&flag,?len)?==?-
            1)?
            ???
            {?
            ??????perror(
            "setsockopt");?
            ??????exit(
            1);?
            ???}
            ?
            ???
            if(?bind(listenfd,(struct?sockaddr*)&servaddr,sizeof(servaddr))?==?
            -1)?
            ???
            {?
            ??????perror(
            "bind");?
            ??????exit(
            1);?
            ???}
            ?
            ???
            else?
            ??????printf(
            "bind?call?OK!\n");?
            ???
            if(?listen(listenfd,5)?==?-1)?
            ???
            {?
            ??????perror(
            "listen");?
            ??????exit(
            1);?
            ???}
            ?
            ???
            for(;;)?
            ???
            {?
            ??????
            if(?(connfd=accept(listenfd,(struct?sockaddr*)NULL,NULL))?==?-1)

            ??????
            {?
            ??????????perror(
            "accept");?
            ??????????exit(
            1);?
            ??????}
            ?
            ??????
            if(?fork()?==?0)/*child?process*/?
            ??????
            {?
            ????????close(listenfd);
            /*這句不能少,原因請大家想想就知道了。*/?
            ????????ticks
            =time(NULL);?
            ????????snprintf(buff,
            100,"%.24s\r\n",ctime(&ticks));?
            ????????write(connfd,buff,strlen(buff));?
            ????????close(connfd);?
            ????????sleep(
            1);?
            ????????execlp(
            "f1-9d",NULL);?
            ????????perror(
            "execlp");?
            ????????exit(
            1);?
            ?????}
            ?
            ?????close(connfd);?
            ?????exit(
            0);/*?end?parent*/?
            ??}
            ?
            }
            ?
            測試:編譯為f1-9d程序,放到一個(gè)自己PATH環(huán)境變量里的某個(gè)路徑里,例如$HO
            ME/bin,運(yùn)行f1-9d,然后telnet?localhost?10013看結(jié)果。

            2、第二種情況我沒有環(huán)境測,所以就不給測試程序了,大家有條件的可以自己寫
            一個(gè)來測試一下。
            ????
            3、測試第三種情況的程序
            #include?<netinet/in.h>?
            #include?
            <sys/socket.h>?
            #include?
            <time.h>?
            #include?
            <stdio.h>?
            #include?
            <string.h>?

            #define?MAXLINE?100?

            int?main(int?argc,?char**?argv)?
            {?
            ???
            int?fd1,fd2;?
            ???
            struct?sockaddr_in?servaddr1,servaddr2;?
            ???
            char?buff[MAXLINE+1];?
            ???time_t?ticks;?
            ???unsigned?
            short?port;?
            ???
            int?flag=1,len=sizeof(int);?

            ???port
            =10013;?
            ???
            if(?(fd1=socket(AF_INET,SOCK_STREAM,0))?==?-1)?
            ???
            {?
            ???????perror(
            "socket");?
            ???????exit(
            1);?
            ???}
            ?
            ???
            if(?(fd2=socket(AF_INET,SOCK_STREAM,0))?==?-1)?
            ???
            {?
            ???????perror(
            "socket");?
            ???????exit(
            1);?
            ???}
            ?
            ???bzero(
            &servaddr1,sizeof(servaddr1));?
            ???bzero(
            &servaddr2,sizeof(servaddr2));?
            ???servaddr1.sin_family
            =AF_INET;?
            ???servaddr2.sin_family
            =AF_INET;?

            ???
            if(?inet_pton(AF_INET,?"127.0.0.1",?&servaddr1.sin_addr)?<=?0)?
            {?
            printf(
            "inet_pton()?call?error:127.0.0.1\n");?
            exit(
            1);?
            }
            ?
            if(?inet_pton(AF_INET,?"128.160.1.230",?&servaddr2.sin_addr)?<=?0)?

            {?
            printf(
            "inet_pton()?call?error:128.160.1.230\n");?
            exit(
            1);?
            }
            ?
            servaddr1.sin_port
            =htons(port);?
            servaddr2.sin_port
            =htons(port);?
            if(?setsockopt(fd1,?SOL_SOCKET,?SO_REUSEADDR,?&flag,?len)?==?-1)?
            {?
            perror(
            "setsockopt");?
            exit(
            1);?
            }
            ?
            if(?setsockopt(fd2,?SOL_SOCKET,?SO_REUSEADDR,?&flag,?len)?==?-1)?
            {?
            perror(
            "setsockopt");?
            exit(
            1);?
            }
            ?
            if(?bind(fd1,(struct?sockaddr*)&servaddr1,sizeof(servaddr1))?==?-1)

            {?
            perror(
            "bind?fd1");?
            exit(
            1);?
            }
            ?
            if(?bind(fd2,(struct?sockaddr*)&servaddr2,sizeof(servaddr2))?==?-1)

            {?
            perror(
            "bind?fd2");?
            exit(
            1);?
            }
            ?
            printf(
            "bind?fd1?and?fd2?OK!\n");?
            /*put?other?process?here*/?
            getchar();?
            exit(
            0);/*?end?*/?
            }
            ?

            4、由于第四種情況只用于UDP的多播,和TCP的使用沒多大關(guān)系,所以就不寫測試
            例子了。自己有興趣的可以寫。

            以上的程序都是在Linux下編譯通過的。也可以在其他unix平臺(tái)運(yùn)行。

            參考資料:
            1、《Unix網(wǎng)絡(luò)編程》卷一 R. Stevens
            2、《Socket programming FAQ》 http://www.ibrado.com/sock-faq/
            posted on 2006-04-29 09:40 Stone Jiang 閱讀(24391) 評論(0)  編輯 收藏 引用 所屬分類: ACEC++&OOP
            色婷婷久久久SWAG精品| 久久亚洲精品成人AV| 日韩人妻无码精品久久免费一| 久久久精品免费国产四虎| 久久国产欧美日韩精品| 久久综合九色综合网站| 日产精品久久久久久久性色| 欧美亚洲国产精品久久| 亚洲人成无码网站久久99热国产| 亚洲伊人久久成综合人影院| 久久亚洲国产成人精品无码区| 久久久久亚洲AV成人网| 欧美一区二区久久精品| 无码人妻久久一区二区三区免费丨 | 久久91精品国产91| 亚洲国产精品综合久久网络| 亚洲а∨天堂久久精品| 超级97碰碰碰碰久久久久最新| 欧美黑人激情性久久| 97热久久免费频精品99| 青青国产成人久久91网| 久久久久人妻精品一区三寸蜜桃| 2021国内精品久久久久久影院| 亚洲αv久久久噜噜噜噜噜| 2021久久精品国产99国产精品| 国产成人精品综合久久久| 亚洲国产婷婷香蕉久久久久久| 久久综合噜噜激激的五月天| 一本一道久久精品综合| 亚洲国产成人久久综合一区77| 日韩人妻无码精品久久久不卡 | 国产精品美女久久久久久2018| 久久综合狠狠综合久久激情 | 亚洲国产成人久久笫一页| 久久国产精品-久久精品| 日日狠狠久久偷偷色综合免费| 久久综合久久自在自线精品自 | 国产精品美女久久久| 国产一区二区久久久| 国产精品免费久久| 日本久久久久亚洲中字幕|