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值