• <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>
            隨筆-159  評(píng)論-223  文章-30  trackbacks-0
                socket pair,也稱套接字管道,主要用來(lái)實(shí)現(xiàn)進(jìn)程內(nèi)或進(jìn)程間的一對(duì)一的全雙工或半雙工通信,在IO復(fù)用模型(如select,poll,epoll等)中起到通知中斷退出循環(huán)的作用,在類UNIX系統(tǒng)中已經(jīng)有現(xiàn)成的實(shí)現(xiàn),API為socketpair,但在Windows系統(tǒng)中沒(méi)有,因此本文主要講述Windows平臺(tái)下soketpair的實(shí)現(xiàn)及應(yīng)用,支持IPv4和IPv6下的tcp、udp套接字管道。
               對(duì)tcp的實(shí)現(xiàn)原理是一端在回環(huán)地址和某端口上監(jiān)聽(tīng)接受另一端的連接;而udp的實(shí)現(xiàn)原理是先在兩端各自綁定回環(huán)地址和某端口,然后設(shè)定對(duì)端地址(調(diào)用connect實(shí)現(xiàn))。綁定的回環(huán)地址在IPv4和IPv6下分別是127.0.0.10:0:0:0:0:0:0:1,而端口由系統(tǒng)分配。這里的實(shí)現(xiàn)具有以下特點(diǎn):
               ● unix僅支持af_unix或af_local地址族,windows僅支持af_init和af_init6地址族。
               ● 僅限于進(jìn)程內(nèi)或父子進(jìn)程間通信,需文件系統(tǒng)路徑名的機(jī)制以支持無(wú)關(guān)進(jìn)程間的通信。
               ● unix使用它作進(jìn)程間通信比標(biāo)準(zhǔn)套接字高效。

            接口
               socketpair創(chuàng)建成功返回0,否則返回-1。
             1#ifdef WIN32
             2#include <winsock2.h>
             3#pragma comment(lib,"ws2_32.lib")
             4#endif
             5
             6#ifdef WIN32
             7typedef SOCKET socket_t;
             8int socketpair(int family,int type,int protocol,SOCKET sock[2]);
             9#else
            10typedef int socket_t;
            11#include <sys/socket.h>
            12#endif

            實(shí)現(xiàn)
               字節(jié)流類型由socketpair_stream實(shí)現(xiàn),數(shù)據(jù)報(bào)類型由socketpair_dgram實(shí)現(xiàn)。
              1#ifdef WIN32
              2#include <ws2tcpip.h>
              3
              4static int socketpair_stream(struct addrinfo* ai,SOCKET sock[2])
              5{
              6    SOCKET listener,client = INVALID_SOCKET,server = INVALID_SOCKET;
              7    int opt = 1;
              8
              9    listener = socket(ai->ai_family,ai->ai_socktype,ai->ai_protocol);
             10    if (INVALID_SOCKET==listener)
             11        goto fail;
             12
             13    setsockopt(listener,SOL_SOCKET,SO_REUSEADDR,(const char*)&opt, sizeof(opt)); 
             14
             15    if(SOCKET_ERROR==bind(listener,ai->ai_addr,ai->ai_addrlen))
             16        goto fail;
             17
             18    if (SOCKET_ERROR==getsockname(listener,ai->ai_addr,(int*)&ai->ai_addrlen))
             19        goto fail;
             20
             21    if(SOCKET_ERROR==listen(listener,SOMAXCONN))
             22        goto fail;
             23
             24    client = socket(ai->ai_family,ai->ai_socktype,ai->ai_protocol);
             25    if (INVALID_SOCKET==client)
             26        goto fail;
             27
             28    if (SOCKET_ERROR==connect(client,ai->ai_addr,ai->ai_addrlen))
             29        goto fail;
             30
             31    server = accept(listener,0,0);
             32    if (INVALID_SOCKET==server)
             33        goto fail;
             34
             35    closesocket(listener);
             36    sock[0= client, sock[1= server;
             37    return 0;
             38
             39fail:
             40    if(INVALID_SOCKET!=listener)
             41        closesocket(listener);
             42    if (INVALID_SOCKET!=client)
             43        closesocket(client);
             44    return -1;
             45}

             46
             47static int socketpair_dgram(struct addrinfo* ai,SOCKET sock[2])
             48{
             49    SOCKET client = INVALID_SOCKET,server=INVALID_SOCKET;
             50    struct addrinfo addr,*res = NULL;
             51    const char* address;
             52    int opt = 1;
             53
             54    server = socket(ai->ai_family,ai->ai_socktype,ai->ai_protocol);
             55    if (INVALID_SOCKET==server)
             56        goto fail;
             57
             58    setsockopt(server,SOL_SOCKET,SO_REUSEADDR,(const char*)&opt, sizeof(opt)); 
             59
             60    if(SOCKET_ERROR==bind(server,ai->ai_addr,ai->ai_addrlen))
             61        goto fail;
             62
             63    if (SOCKET_ERROR==getsockname(server,ai->ai_addr,(int*)&ai->ai_addrlen))
             64        goto fail;
             65
             66    client = socket(ai->ai_family,ai->ai_socktype,ai->ai_protocol);
             67    if (INVALID_SOCKET==client)
             68        goto fail;
             69
             70    memset(&addr,0,sizeof(addr));
             71    addr.ai_family = ai->ai_family;
             72    addr.ai_socktype = ai->ai_socktype;
             73    addr.ai_protocol = ai->ai_protocol;
             74
             75    if (AF_INET6==addr.ai_family)
             76        address = "0:0:0:0:0:0:0:1";
             77    else
             78        address = "127.0.0.1";
             79
             80    if (getaddrinfo(address,"0",&addr,&res))
             81        goto fail;
             82
             83    setsockopt(client,SOL_SOCKET,SO_REUSEADDR,(const char*)&opt, sizeof(opt)); 
             84    if(SOCKET_ERROR==bind(client,res->ai_addr,res->ai_addrlen))
             85        goto fail;
             86
             87    if (SOCKET_ERROR==getsockname(client,res->ai_addr,(int*)&res->ai_addrlen))
             88        goto fail;
             89
             90    if (SOCKET_ERROR==connect(server,res->ai_addr,res->ai_addrlen))
             91        goto fail;
             92
             93    if (SOCKET_ERROR==connect(client,ai->ai_addr,ai->ai_addrlen))
             94        goto fail;
             95
             96    freeaddrinfo(res);
             97    sock[0= client, sock[1= server;
             98    return 0;
             99
            100fail:
            101    if (INVALID_SOCKET!=client)
            102        closesocket(client);
            103    if (INVALID_SOCKET!=server)
            104        closesocket(server);
            105    if (res)
            106        freeaddrinfo(res);
            107    return -1;
            108}

            109
            110int socketpair(int family,int type,int protocol,SOCKET sock[2])
            111{
            112    const char* address;
            113    struct addrinfo addr,*ai;
            114    int ret = -1;
            115
            116    memset(&addr,0,sizeof(addr));
            117    addr.ai_family = family;
            118    addr.ai_socktype = type;
            119    addr.ai_protocol = protocol;
            120
            121    if (AF_INET6==family)
            122        address = "0:0:0:0:0:0:0:1";
            123    else
            124        address = "127.0.0.1";
            125
            126    if (0==getaddrinfo(address,"0",&addr,&ai))
            127    {
            128        if (SOCK_STREAM==type)
            129            ret = socketpair_stream(ai,sock);
            130        else if(SOCK_DGRAM==type)
            131            ret = socketpair_dgram(ai,sock); 
            132        freeaddrinfo(ai);
            133    }

            134    return ret;
            135}

            138#endif

            應(yīng)用
             1    //ipv4字節(jié)流套接字管道
             2    ret = socketpair(AF_INET,SOCK_STREAM,0,sock);
             3    if (-1==ret) return -1;
             4
             5    ret = send(sock[0],"ipv4 tcp: hello sock 1\n",24,0);
             6    ret = recv(sock[1],buf,sizeof(buf),0);
             7    OutputDebugStringA(buf);
             8    
             9    ret = send(sock[1],"ipv4 tcp: hello sock 0\n",24,0);
            10    ret = recv(sock[0],buf,sizeof(buf),0);
            11    OutputDebugStringA(buf);
            12
            13    //ipv4數(shù)據(jù)報(bào)套接字管道
            14    ret = socketpair(AF_INET,SOCK_DGRAM,0,sock);
            15    if (-1==ret) return -1;
            16
            17    ret = send(sock[0],"ipv4 udp: hello sock 1\n",24,0);
            18    ret = recv(sock[1],buf,sizeof(buf),0);
            19    OutputDebugStringA(buf);
            20
            21    ret = sendto(sock[1],"ipv4 udp: hello sock 0\n",24,0,NULL,0);
            22    ret = recvfrom(sock[0],buf,sizeof(buf),0,(struct sockaddr*)&r_addr,&r_len);
            23    OutputDebugStringA(buf);
            24
            25    //ipv6字節(jié)流套接字管道
            26    ret = socketpair(AF_INET6,SOCK_STREAM,IPPROTO_TCP,sock);
            27    if (-1==ret) return -1;
            28
            29    ret = send(sock[0],"ipv6 tcp: hello sock 1\n",24,0);
            30    ret = recv(sock[1],buf,sizeof(buf),0);
            31    OutputDebugStringA(buf);
            32
            33    ret = send(sock[1],"ipv6 tcp: hello sock 0\n",24,0);
            34    ret = recv(sock[0],buf,sizeof(buf),0);
            35    OutputDebugStringA(buf);
            36
            37    //ipv6數(shù)據(jù)報(bào)套接字管道
            38    ret = socketpair(AF_INET6,SOCK_DGRAM,IPPROTO_UDP,sock);
            39    if (-1==ret) return -1;
            40
            41    ret = send(sock[0],"ipv6 udp: hello sock 1\n",24,0);
            42    ret = recv(sock[1],buf,sizeof(buf),0);
            43    OutputDebugStringA(buf);
            44
            45    ret = sendto(sock[1],"ipv6 udp: hello sock 0\n",24,0,NULL,0);
            46    ret = recvfrom(sock[0],buf,sizeof(buf),0,NULL,0);
            47    OutputDebugStringA(buf);
               從上可得,對(duì)于已連接的udp套接字,可調(diào)用recv或recvfrom接收數(shù)據(jù),調(diào)用send或sendto發(fā)送數(shù)據(jù),但調(diào)用sendto則不能指定目標(biāo)地址。測(cè)試主機(jī)需支持IPv4和IPv6雙協(xié)議棧,輸出如下:
            ipv4 tcp: hello sock 1
            ipv4 tcp: hello sock 
            0
            ipv4 udp: hello sock 
            1
            ipv4 udp: hello sock 
            0
            ipv6 tcp: hello sock 
            1
            ipv6 tcp: hello sock 
            0
            ipv6 udp: hello sock 
            1
            ipv6 udp: hello sock 
            0
            posted on 2012-06-17 03:02 春秋十二月 閱讀(3026) 評(píng)論(3)  編輯 收藏 引用 所屬分類: Network

            評(píng)論:
            # re: 一種簡(jiǎn)單的跨平臺(tái)套接字管道 2012-06-18 13:30 | HASKELL
            非父子進(jìn)程間通信無(wú)法實(shí)現(xiàn)。
              回復(fù)  更多評(píng)論
              
            # re: 一種簡(jiǎn)單的跨平臺(tái)套接字管道 2015-11-05 13:40 | IT搬運(yùn)工
            windows僅支持af_init和af_init6地址族有錯(cuò)別字么?

            af_init和af_init6  回復(fù)  更多評(píng)論
              
            # re: 一種簡(jiǎn)單的跨平臺(tái)套接字管道 2015-11-05 16:43 | IT搬運(yùn)工
            if (AF_INET6==family)
            address = "0:0:0:0:0:0:0:1";

            為什么不是 ::1 ?

            記得0.0.0.0對(duì)應(yīng)::
            127.0.0.1對(duì)應(yīng)::1  回復(fù)  更多評(píng)論
              
            97久久婷婷五月综合色d啪蜜芽| 久久精品无码免费不卡| 久久97久久97精品免视看| 99久久这里只精品国产免费| 国内精品人妻无码久久久影院| 手机看片久久高清国产日韩| 久久亚洲日韩精品一区二区三区| 国产精品丝袜久久久久久不卡| 久久久久亚洲AV无码永不| 国产免费久久精品99re丫y| 久久狠狠高潮亚洲精品| 91精品国产91热久久久久福利 | 久久精品无码午夜福利理论片| 久久香蕉一级毛片| 九九久久99综合一区二区| 国产亚洲精久久久久久无码| 久久久久免费视频| 久久精品一区二区三区不卡| 狠狠色综合网站久久久久久久高清| 狠狠色丁香婷综合久久| 少妇人妻88久久中文字幕| 亚洲午夜久久久影院| 精品久久人人做人人爽综合 | 三级三级久久三级久久| 色综合久久精品中文字幕首页| 亚洲精品无码久久不卡| 国产一区二区久久久| 国产精品美女久久久久av爽 | 久久精品无码专区免费 | 精品国产综合区久久久久久| 亚洲AV日韩AV永久无码久久| 中文字幕无码久久人妻| 亚洲精品NV久久久久久久久久| 久久这里只精品国产99热| 国产精品国色综合久久| 久久亚洲美女精品国产精品| 久久香蕉国产线看观看精品yw | 久久99精品综合国产首页| 久久99亚洲网美利坚合众国| 久久国产亚洲精品无码| 久久97精品久久久久久久不卡|