• <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久久99小草精品免视看| 久久青青草原精品国产| 久久国产福利免费| 国产成人精品免费久久久久| 国产人久久人人人人爽| 中文国产成人精品久久不卡| 一本久道久久综合狠狠躁AV| 人妻中文久久久久| 国产精品一区二区久久精品涩爱| 中文字幕精品久久久久人妻| 国产成人精品久久亚洲高清不卡| 国内精品久久久久久久影视麻豆 | 久久只有这里有精品4| 国产亚洲成人久久| 久久免费99精品国产自在现线| 中文成人无码精品久久久不卡| 久久99精品久久久久久噜噜| 午夜精品久久影院蜜桃| 国产毛片欧美毛片久久久| 久久精品aⅴ无码中文字字幕重口| 色综合久久久久综合体桃花网| 亚洲欧美精品伊人久久| 一个色综合久久| AV无码久久久久不卡蜜桃| 69国产成人综合久久精品| 亚洲国产精品成人久久蜜臀 | 一本一本久久A久久综合精品| 亚洲AV无码久久精品蜜桃| 高清免费久久午夜精品| 免费一级欧美大片久久网| 免费精品国产日韩热久久| 国内精品久久久久久99| 精品久久人人妻人人做精品| 久久经典免费视频| 2021少妇久久久久久久久久| 伊人 久久 精品| 久久综合久久综合久久综合| 久久人人爽爽爽人久久久| 久久久久久久91精品免费观看| 久久精品国产亚洲AV高清热|