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

            接口
               socketpair創建成功返回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

            實現
               字節流類型由socketpair_stream實現,數據報類型由socketpair_dgram實現。
              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

            應用
             1    //ipv4字節流套接字管道
             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數據報套接字管道
            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字節流套接字管道
            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數據報套接字管道
            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);
               從上可得,對于已連接的udp套接字,可調用recv或recvfrom接收數據,調用send或sendto發送數據,但調用sendto則不能指定目標地址。測試主機需支持IPv4和IPv6雙協議棧,輸出如下:
            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) 評論(3)  編輯 收藏 引用 所屬分類: Network

            評論:
            # re: 一種簡單的跨平臺套接字管道 2012-06-18 13:30 | HASKELL
            非父子進程間通信無法實現。
              回復  更多評論
              
            # re: 一種簡單的跨平臺套接字管道 2015-11-05 13:40 | IT搬運工
            windows僅支持af_init和af_init6地址族有錯別字么?

            af_init和af_init6  回復  更多評論
              
            # re: 一種簡單的跨平臺套接字管道 2015-11-05 16:43 | IT搬運工
            if (AF_INET6==family)
            address = "0:0:0:0:0:0:0:1";

            為什么不是 ::1 ?

            記得0.0.0.0對應::
            127.0.0.1對應::1  回復  更多評論
              
            奇米影视7777久久精品| 久久精品男人影院| 久久国产精品99国产精| 久久国产高清字幕中文| 久久se精品一区精品二区国产| 人妻无码精品久久亚瑟影视| 色偷偷久久一区二区三区| 国产精品成人99久久久久91gav| 亚洲精品国产第一综合99久久 | 爱做久久久久久| 狠狠色综合网站久久久久久久高清 | 久久av无码专区亚洲av桃花岛| 91久久福利国产成人精品| 一本色综合网久久| 久久综合精品国产一区二区三区 | 久久人人爽人人爽人人片AV东京热| 狠狠色婷婷久久一区二区三区| 亚洲国产成人精品女人久久久| 九九99精品久久久久久| 亚洲精品午夜国产VA久久成人| 国产午夜精品久久久久九九| 久久人人妻人人爽人人爽| 亚洲国产成人精品无码久久久久久综合 | 亚洲精品tv久久久久久久久久| 国产精品久久国产精麻豆99网站| 久久亚洲精品成人无码网站| 国产精品成人无码久久久久久| 99久久99久久精品国产片果冻 | 久久久国产视频| 青青草国产97免久久费观看| 久久99国产精品99久久| 久久国产乱子伦免费精品| 囯产精品久久久久久久久蜜桃| 亚洲国产成人久久一区久久| 国产精品热久久毛片| 9191精品国产免费久久| 婷婷综合久久中文字幕| 香蕉久久夜色精品国产小说| 精品久久久久久亚洲精品| 久久精品麻豆日日躁夜夜躁| 日产精品久久久一区二区|