搜索
您的当前位置:首页正文

ntohs, ntohl, htons,htonl的比较和详解

来源:易榕旅网
在C/C++写网络程序的时候,往往会遇到字节的网络顺序和主机顺序的问题。
这时就可能用到htons(), ntohl(), ntohs(),htons()这4个网络字节顺序与本地字节顺序之间的转换函数:

之所以需要这些函数是因为计算机数据表示存在两种字节顺序:NBO与HBO。

网络字节顺序NBO(Network Byte Order):
      按从高到低的顺序存储,在网络上使用统一的网络字节顺序,可以避免兼容性问题。

主机字节顺序(HBO,Host Byte Order):
      不同的机器HBO不相同,与CPU设计有关,数据的顺序是由cpu决定的,而与操作系统无关。 
      如 Intel   x86结构下,short型数0x1234表示为34 12, int型数0x12345678表示为78 56 34 12 。
      如IBM   power PC结构下,short型数0x1234表示为12   34, int型数0x12345678表示为12 34 56 78。   

 由于这个原因不同体系结构的机器之间无法通信,所以要转换成一种约定的数序,也就是网络字节顺序,其实就是如同power   pc那样的顺序 。在PC开发中有ntohl和htonl函数可以用来进行网络字节和主机字节的转换。


在Linux系统下:htonl(),htons(), ntohl(), ntohs()的头文件及函数定义:
  #include <arpa/inet.h>
  uint32_t htonl(uint32_t hostlong);
  uint16_t htons(uint16_t hostshort);
  uint32_t ntohl(uint32_t netlong);
  uint16_t ntohs(uint16_t netshort);

在windows系统下htonl(),htons(), ntohl(), ntohs(), inet_addr()使用说明

ntohs()  

  简述:
  将一个无符号短整形数从网络字节顺序转换为主机字节顺序。
  #include <winsock.h>
  u_short PASCAL FAR ntohs( u_short netshort);
  netshort:一个以网络字节顺序表达的16位数。
  注释:
  本函数将一个16位数由网络字节顺序转换为主机字节顺序。
  返回值:ntohs()返回一个以主机字节顺序表达的数。

ntohl() 

       简述:
  将一个无符号长整形数从网络字节顺序转换为主机字节顺序。
  #include <winsock.h>
  u_long PASCAL FAR ntohl( u_long netlong);
  netlong:一个以网络字节顺序表达的32位数。
  注释:
  本函数将一个32位数由网络字节顺序转换为主机字节顺序。
  返回值:
  ntohl()返回一个以主机字节顺序表达的数。

htons()
       简述:
  将主机的无符号短整形数转换成网络字节顺序。//将无符号短整型主机字节序转换为网络字节序
  #include <winsock.h>
  u_short PASCAL FAR htons( u_short hostshort);
  hostshort:主机字节顺序表达的16位数。
  注释:
  本函数将一个16位数从主机字节顺序转换成网络字节顺序。
  返回值:
  htons()返回一个网络字节顺序的值。
  简单地说,htons()就是将一个数的高低位互换
  (如:12 34 --> 34 12)
  VB表示:
  MsgBox Hex(htons(&H1234))
  显示值为 3412

htonl()
      简述:
  将主机的无符号长整形数转换成网络字节顺序。//将无符号长整型网络字节序转换为主机字节序
  #include <winsock.h>
  u_long PASCAL FAR htonl( u_long hostlong);
  hostlong:主机字节顺序表达的32位数。
  注释:
  本函数将一个32位数从主机字节顺序转换成网络字节顺序。
  返回值:
  htonl()返回一个网络字节顺序的值。

inet_addr()

inet_aton()
    与inet_ntoa()作用相反。

atoi()

array to integer将字符串转换为整形数



***********************************************************************************************************************************

测试代码如下

#include 
#include 
#include 
#include 
#include 
int main(int argc, char* argv[])
{
         struct in_addr addr1,addr2;
         ulong   l1,l2;
         l1= inet_addr("192.168.0.74");
         l2 = inet_addr("211.100.21.179");
         memcpy(&addr1, &l1, 4);
         memcpy(&addr2, &l2, 4);

         printf("%s : %s ", inet_ntoa(addr1), inet_ntoa(addr2));    //注意这一句的运行结果

         printf("%s ", inet_ntoa(addr1));
         printf("%s ", inet_ntoa(addr2));
         return 0;
}
实际运行结果如下:
192.168.0.74 : 192.168.0.74       //从这里可以看出,printf里的inet_ntoa只运行了一次。
192.168.0.74
211.100.21.179

inet_ntoa返回一个char *,而这个char *的空间是在inet_ntoa里面静态分配的,所以inet_ntoa后面的调用会覆盖上一次的调用。第一句printf的结果只能说明在printf里面的可变参数的求值是从右到左的,仅此而已。

因篇幅问题不能全部显示,请点此查看更多更全内容

Top