1、htonl ()和ntohl( )
u_long PASCAL FAR ntohl (u_long netlong);
u_short PASCAL FAR ntohs (u_short netshort);
ntohl( )-----網絡順序轉換成主機順序
u_long PASCAL FAR htonl (u_long hostlong);
u_short PASCAL FAR htons (u_short hostshort);
htonl ()-----主機順序轉換成網絡順序
2、inet_addr( )和inet_ntoa ( )
unsigned long PASCAL FAR inet_addr (const char FAR * cp);
char FAR * PASCAL FAR inet_ntoa (struct in_addr in);
inet_addr函數需要一個字符串作為其參數,該字符串指定了以點分十進制格式表示的IP地址(例如:192.168.0.16)。而且inet_addr函數會返回一個適合分配給S_addr的u_long類型的數值。
Inet_ntoa函數會完成相反的轉換,它接受一個in_addr結構體類型的參數並返回一個以點分十進制格式表示的IP地址字符串。
struct sockaddr {
unsigned short sa_family;
char sa_data[14];
};
上面是通用的socket地址,具體到Internet socket,用下面的結構,二者可以進行類型轉換
struct sockaddr_in {
short int sin_family;
unsigned short int sin_port;
struct in_addr sin_addr;
unsigned char sin_zero[8];
};
struct in_addr就是32位IP地址。
struct in_addr {
union {
struct { u_char s_b1,s_b2,s_b3,s_b4; } S_un_b;
struct { u_short s_w1,s_w2; } S_un_w;
u_long S_addr;
} S_un;
#define s_addr S_un.S_addr
};
inet_addr()是將一個點分制的IP地址(如192.168.0.1)轉換為上述結構中需要的32位IP地址(0xC0A80001)。
填值的時候使用sockaddr_in結構,而作為函數(如socket, listen, bind等)的參數傳入的時候轉換成sockaddr結構就行了,畢竟都是16個字符長。
通常的用法是:
int sockfd;
struct sockaddr_in my_addr;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons(MYPORT);
my_addr.sin_addr.s_addr = inet_addr("192.168.0.1");
bzero(&(my_addr.sin_zero), 8);
bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));
可以用C++做個不太准確的假設。
sockaddr是base class
sockaddr_in 等是derived class
如此一來,bind, connect , sendto , recvfrom等函數就可以使用base class
來處理多種不同的derived class了。
但是實際上,這是沒有繼承關系數據結構(C嘛),所以需要強制造型來轉換數據類型。正因為如此,在sendto的時候需要給出len長度,因為不同的sockaddr_xx實現長度並不相同。
名詞解析:
主機字節序:
不同的CPU有不同的字節序類型,這些字節序是指整數在內存中保存的順序,這個叫做主機序。最常見的有兩種 1.Little endian:低字節存高地址,高字節存低地址 2.Big endian:低字節存低地址,高字節存高地址
網絡字節序:
網絡字節順序是TCP/IP中規定好的一種數據表示格式,它與具體的CPU類型、操作系統等無關,從而可以保證數據在不同主機之間傳輸時能夠被正確解釋。網絡字節順序采用big endian排序方式。
為了進行轉換bsd socket提供了轉換的函數,有下面四個網絡與主機字節轉換函數:htons ntohs htonl ntohl (s 就是short l是long h是host n是network)
htons 把unsigned short類型從主機序轉換到網絡序,htonl 把unsigned long類型從主機序轉換到網絡序,ntohs 把unsigned short類型從網絡序轉換到主機序,ntohl 把unsigned long類型從網絡序轉換到主機序。
在使用little endian的系統中 這些函數會把字節序進行轉換 在使用big endian類型的系統中這些函數會定義成空宏。
將用點分割的IP地址轉換位一個in_addr結構的地址,這個結構的定義見筆記(一),實際上就是一個unsigned long值。計算機內部處理IP地址可是不認識如192.1.8.84之類的數據。
unsigned long inet_addr( const char FAR * cp );
舉例:inet_addr("192.1.8.84")=1409810880
inet_addr("127.0.0.1")= 16777343
如果發生錯誤,函數返回INADDR_NONE值。
將網絡地址轉換位用點分割的IP地址,是上面函數的逆函數。
char FAR * inet_ntoa( struct in_addr in );
舉例:char * ipaddr=NULL;
char addr[20];
in_addr inaddr;
inaddr. s_addr=16777343;
ipaddr= inet_ntoa(inaddr);
strcpy(addr,ipaddr);
這樣addr的值就變為127.0.0.1。
注意不要修改返回值或者進行釋放動作,如果函數失敗就會返回NULL值。
