• <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>
            隨筆-161  評論-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 春秋十二月 閱讀(3069) 評論(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  回復  更多評論
              
            久久久亚洲欧洲日产国码二区| 午夜视频久久久久一区| 亚洲精品国产自在久久| 欧美久久久久久| 综合人妻久久一区二区精品| 久久丫精品国产亚洲av不卡| 一级做a爰片久久毛片16| 久久久久无码中| 精品久久久无码21p发布| 99久久国产综合精品成人影院| 久久九九久精品国产免费直播| 99久久综合国产精品免费| 一级做a爰片久久毛片16| 亚洲国产精品无码久久久久久曰| 国产精品久久久香蕉| 久久精品一区二区| 亚洲va久久久噜噜噜久久狠狠 | 久久国产精品无码一区二区三区| 欧美综合天天夜夜久久| 亚洲AV成人无码久久精品老人| 欧美777精品久久久久网| 亚洲狠狠婷婷综合久久蜜芽 | 久久婷婷五月综合色奶水99啪| 精品久久8x国产免费观看| 久久久一本精品99久久精品66 | 精品久久久久久久中文字幕 | 青青青国产成人久久111网站| 精品国产乱码久久久久久人妻 | 精品乱码久久久久久夜夜嗨| 久久久久99精品成人片欧美 | 香蕉久久av一区二区三区| 久久精品二区| 久久久久黑人强伦姧人妻| 久久青青草原精品影院| 国产成人精品白浆久久69| 国产精品一久久香蕉国产线看观看| 一本大道久久香蕉成人网| 亚洲国产成人精品91久久久 | 99久久婷婷国产综合亚洲| 国产午夜精品久久久久免费视| 久久夜色精品国产噜噜麻豆|