UNIX網路程式設計學習(9)--getsockname和getpeername的用法及例項

pengfoo發表於2012-03-25
getsockname和getpeername

#include <sys/socket.h>
int getsockname(int sockfd, struct sockaddr *localaddr, socklen_t *addrlen);
int getpeername(int sockfd, struct sockaddr *peeraddr, socklen_t *addrlen);
返回:0—OK,-1—出錯。

getsockname函式返回與套介面關聯的本地協議地址。

getpeername函式返回與套介面關聯的遠端協議地址。

addrlen是值-結果引數。

使用場合:

  • 在不呼叫bind的TCP客戶,當connect成功返回後,getsockname返回分配給此連線的本地IP地址和本地埠號;
  • 在以埠號為0呼叫bind後,使用getsockname返回核心分配的本地埠號;
  • getsockname可用來獲取某套介面的地址族;
  • 在捆綁了通配IP地址的TCP伺服器上,當連線建立後,可以使用getsockname獲得分配給此連線的本地IP地址;
  • 當一個伺服器呼叫exec啟動後,他獲得客戶身份的唯一途徑是呼叫getpeername函式。

伺服器端原始碼:

#include	"unp.h"

int
main(int argc, char ** argv)
{
    int         listenfd,connfd;
    struct      sockaddr_in servaddr;
    pid_t       pid;
    char        temp[20];

    listenfd = Socket(AF_INET, SOCK_STREAM, 0);
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(10010);
    Bind(listenfd, (SA *)&servaddr, sizeof(servaddr));
    Listen(listenfd, LISTENQ);
    for( ; ; )
    {
        struct sockaddr_in local;
        connfd = Accept(listenfd, (SA *)NULL, NULL);
        if((pid = fork()) == 0)
        {
            Close(listenfd);struct sockaddr_in serv, guest;
            char serv_ip[20];
            char guest_ip[20];
            socklen_t serv_len = sizeof(serv);
            socklen_t guest_len = sizeof(guest);
            getsockname(connfd, (struct sockaddr *)&serv, &serv_len);
            getpeername(connfd, (struct sockaddr *)&guest, &guest_len);
            Inet_ntop(AF_INET, &serv.sin_addr, serv_ip, sizeof(serv_ip));
            Inet_ntop(AF_INET, &guest.sin_addr, guest_ip, sizeof(guest_ip));
            printf("host %s:%d guest %s:%d\n", serv_ip, ntohs(serv.sin_port), guest_ip, ntohs(guest.sin_port));
            char buf[] = "hello world";
            Write(connfd, buf, strlen(buf));
            Close(connfd);
            exit(0);
        }
        Close(connfd);
    }
}


客戶端原始碼:

#include "unp.h"
#define DEST_IP "127.0.0.1"

int
main(int argc, char ** argv)
{
    int         sockfd, n;
    char        buf[100];
    char        serv_ip[20], guest_ip[20];
    struct      sockaddr_in servaddr;

    sockfd = Socket(AF_INET, SOCK_STREAM, 0);
    bzero(&servaddr, sizeof(struct sockaddr_in));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(10010);

    Inet_pton(AF_INET, DEST_IP, &servaddr.sin_addr);
    Connect(sockfd, (SA *)&servaddr, sizeof(servaddr));

    struct sockaddr_in serv, guest;
    socklen_t serv_len = sizeof(serv);
    socklen_t guest_len = sizeof(guest);

    getsockname(sockfd, (SA *)&guest, &guest_len);
    getpeername(sockfd, (SA *)&serv, &serv_len);

    Inet_ntop(AF_INET, &guest.sin_addr, guest_ip, sizeof(guest_ip));
    Inet_ntop(AF_INET, &serv.sin_addr, serv_ip, sizeof(serv_ip));

    printf("host  %s:%d, guest  %s:%d\n", serv_ip, ntohs(serv.sin_port), guest_ip, ntohs(guest.sin_port));

    n = Read(sockfd, buf, 100);
    buf[n] = '\0';
    printf("%s\n", buf);
    Close(sockfd);
    exit(0);
}

TCP

對於伺服器來說,在bind以後就可以呼叫getsockname來獲取本地地址和埠,雖然這沒有什麼太多的意義。getpeername只有在連結建立以後才呼叫,否則不能正確獲得對方地址和埠,所以他的引數描述字一般是連結描述字而非監聽套介面描述字。

對於客戶端來說,在呼叫socket時候核心還不會分配IP和埠,此時呼叫getsockname不會獲得正確的埠和地址(當然連結沒建立更不可能呼叫getpeername),當然如果呼叫了bind 以後可以使用getsockname。想要正確的到對方地址(一般客戶端不需要這個功能),則必須在連結建立以後,同樣連結建立以後,此時客戶端地址和埠就已經被指定,此時是呼叫getsockname的時機。


 

相關文章