【網絡】IP地址格式轉換(htonl、ntohl;inet_addr、inet_ntoa)


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地址字符串。

sockaddr_in , sockaddr , in_addr區別

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值。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM