• <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ī)器可
            以測(cè)試這種情況。
            ????3、SO_REUSEADDR允許單個(gè)進(jìn)程綁定相同的端口到多個(gè)socket上,但每個(gè)soc
            ket綁定的ip地址不同。這和2很相似,區(qū)別請(qǐng)看UNPv1。
            ????4、SO_REUSEADDR允許完全相同的地址和端口的重復(fù)綁定。但這只用于UDP的
            多播,不用于TCP。

            例子1:測(cè)試上面第一種情況。
            #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);
            /*這句不能少,原因請(qǐng)大家想想就知道了。*/?
            ????????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*/?
            ??}
            ?
            }
            ?
            測(cè)試:編譯為f1-9d程序,放到一個(gè)自己PATH環(huán)境變量里的某個(gè)路徑里,例如$HO
            ME/bin,運(yùn)行f1-9d,然后telnet?localhost?10013看結(jié)果。

            2、第二種情況我沒有環(huán)境測(cè),所以就不給測(cè)試程序了,大家有條件的可以自己寫
            一個(gè)來測(cè)試一下。
            ????
            3、測(cè)試第三種情況的程序
            #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)系,所以就不寫測(cè)試
            例子了。自己有興趣的可以寫。

            以上的程序都是在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 閱讀(24378) 評(píng)論(0)  編輯 收藏 引用 所屬分類: ACEC++&OOP
            青青久久精品国产免费看| 日本三级久久网| 久久久久久久精品妇女99| 一级a性色生活片久久无| 久久青青色综合| 久久婷婷五月综合国产尤物app| 99久久无色码中文字幕人妻| 国产成人久久AV免费| 久久国产精品波多野结衣AV | 精品国产乱码久久久久久呢| 久久99久久99精品免视看动漫| 久久av无码专区亚洲av桃花岛| 韩国三级中文字幕hd久久精品| 久久人做人爽一区二区三区| 国产午夜精品久久久久九九| 精品人妻伦九区久久AAA片69| 色综合久久最新中文字幕| 久久天天躁狠狠躁夜夜躁2014| 国产91久久综合| 国产亚洲精品美女久久久| 一本久久综合亚洲鲁鲁五月天亚洲欧美一区二区| 亚洲AV无码久久精品狠狠爱浪潮| 久久精品一区二区三区中文字幕 | 久久九九免费高清视频| 久久大香香蕉国产| 精品人妻伦九区久久AAA片69| 久久精品国产只有精品66 | 久久久国产打桩机| 国产亚州精品女人久久久久久| 精品熟女少妇a∨免费久久| 久久婷婷五月综合97色直播| 亚洲伊人久久综合中文成人网| 久久久无码精品午夜| 99久久国产免费福利| 青青草国产精品久久| 99久久精品费精品国产一区二区 | 色老头网站久久网| 一级a性色生活片久久无| 欧美日韩精品久久久久| 久久久久亚洲国产| 国产精品中文久久久久久久 |