From: http://www.it165.NET/PRo/html/201211/4066.html
在初學的時候對二者甚是迷茫一會就暈了 在這里總結一句話:
struct sockaddr 這個結構體是linux的網絡編程接口中用來表示ip地址的標準結構體,bind、connect等函數中都需要這個結構體,這個結構體是兼容IPV4和IPV6的。在實際編程中這個結構體會被一個struct sockaddr_in所填充。
sockaddr 在bind的man手冊中提到
struct sockaddr { sa_family_t sa_family; //所選協議族AF_INET char sa_data[14]; //ip地址及端口號 }而sockaddr_in
{ __SOCKADDR_COMMON (sin_); in_port_t sin_port; /* Port number. */ struct in_addr sin_addr; /* Internet address. */ unsigned char sin_zero[sizeof (struct sockaddr) - __SOCKADDR_COMMON_SIZE - sizeof (in_port_t) - sizeof (struct in_addr)]; };雖然是兩個結構體可是二者的占用的內存是一致的,因此可以互相轉化。
sockaddr常用于bind、connect、recvfrom、sendto等函數的參數,指明地址信息。是一種通用的套接字地址。
而sockaddr_in 一般是儲存地址和端口的。用于信息的顯示及存儲使用
例如:
struct sockaddr_in addr_server;addr_server.sin_family = AF_INET;addr_server.sin_port = htons(RPORT);addr_server.sin_addr.s_addr = inet_addr(RHOST);然而,在類似于bind accept的函數中
ret = bind(fd_sock, (struct sockaddr *)&addr_server, sizeof(addr_server));if(ret < 0){ perror("bind"); return -1;}之前只是這樣的記下來了,可是知道一天,想顯示所連接的客戶端的ip地址的時候,就發現了問題所在
char *inet_ntoa(struct in_addr in);函數原型是這樣的,可是在
struct in_addr{ in_addr_t s_addr;};這個in_addr是sockaddr_in的一個mamber
fd_connection = accept(fd_sock, (struct sockaddr *)&addr_client, &addr_client_len); if(fd_connection < 0){ perror("accept"); return -1; } printf("connected! : %d/n", fd_connection); printf("%s%s/n", "the client ip is :", inet_ntoa(addr_client.sin_addr));新聞熱點
疑難解答