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函數。
串連建立成功後,我們就可以像使用網路通訊端一樣進行發送和接受操作了。甚至還可以將串連設定為非阻塞模式,這裡就不贅述了。