sockaddr
struct sockaddr {
unsigned short sa_family; /* address family, AF_xxx */
char sa_data[14]; /* 14 bytes of protocol address */
};
sa_family是地址家族,一般都是“AF_xxx”的形式。好像通常大多用的是都是AF_INET。
sa_data是14字節協議地址。
此數據結構用做bind、connect、recvfrom、sendto等函數的參數,指明地址信息。
但一般編程中並不直接針對此數據結構操作,而是使用另一個與sockaddr等價的數據結構
sockaddr_in
sockaddr_in(在netinet/in.h中定義):
struct sockaddr_in {
short int sin_family; /* Address family */
unsigned short int sin_port; /* Port number */
struct in_addr sin_addr; /* Internet address */
unsigned char sin_zero[8]; /* Same size as struct sockaddr */
};
struct in_addr {
unsigned long s_addr;
};
typedef struct in_addr {
union {
struct{
unsigned char s_b1,
s_b2,
s_b3,
s_b4;
} S_un_b;
struct {
unsigned short s_w1,
s_w2;
} S_un_w;
unsigned long S_addr;
} S_un;
} IN_ADDR;
sin_family指代協議族,在socket編程中只能是AF_INET
sin_port存儲端口號(使用網絡字節順序)
sin_addr存儲IP地址,使用in_addr這個數據結構
sin_zero是為了讓sockaddr與sockaddr_in兩個數據結構保持大小相同而保留的空字節。
s_addr按照網絡字節順序存儲IP地址
sockaddr_in和sockaddr是並列的結構,指向sockaddr_in的結構體的指針也可以指向
sockadd的結構體,並代替它。也就是說,你可以使用sockaddr_in建立你所需要的信息,
在最后用進行類型轉換就可以了bzero((char*)&mysock,sizeof(mysock));//初始化
mysock結構體名
mysock.sa_family=AF_INET;
mysock.sin_addr.s_addr=inet_addr("192.168.0.1");
……
等到要做轉換的時候用:
(struct sockaddr*)mysock
sockaddr_un
進程間通信的一種方式是使用UNIX套接字,人們在使用這種方式時往往用的不是網絡套接字,而是一種稱為本地套接字的方式。這樣做可以避免為黑客留下后門。
創建
使用套接字函數socket創建,不過傳遞的參數與網絡套接字不同。域參數應該是PF_LOCAL或者PF_UNIX,而不能用PF_INET之類。本地套接字的通訊類型應該是SOCK_STREAM或SOCK_DGRAM,協議為默認協議。例如:
int sockfd;
sockfd = socket(PF_LOCAL, SOCK_STREAM, 0);
綁定
創建了套接字后,還必須進行綁定才能使用。不同於網絡套接字的綁定,本地套接字的綁定的是struct sockaddr_un結構。struct sockaddr_un結構有兩個參數:sun_family、sun_path。sun_family只能是AF_LOCAL或AF_UNIX,而sun_path是本地文件的路徑。通常將文件放在/tmp目錄下。例如:
struct sockaddr_un sun;
sun.sun_family = AF_LOCAL;
strcpy(sun.sun_path, filepath);
bind(sockfd, (struct sockaddr*)&sun, sizeof(sun));
監聽
本地套接字的監聽、接受連接操作與網絡套接字類似。
連接
連接到一個正在監聽的套接字之前,同樣需要填充struct sockaddr_un結構,然后調用connect函數。
連接建立成功后,我們就可以像使用網絡套接字一樣進行發送和接受操作了。甚至還可以將連接設置為非阻塞模式,這里就不贅述了。
版權聲明:本文為博主原創文章,未經博主允許不得轉載。
