socket网络编程之htons()相关详解

1、走起

ntohs =net to host short int 16位
htons=host to net short int 16位
ntohl=net to host long int 32位
htonl=host to net long int 32位

Linux系统下,头文件及函数定义如下:

#include

uint32_t htonl(uint32_t hostlong);

uint16_t htons(uint16_t hostshort);

uint32_t ntohl(uint32_t netlong);

uint16_t ntohs(uint16_t netshort);

网络字节顺序NBO(Network Byte Order)

按从高到低的顺序存储,在网络上使用同一的网络字节顺序,可避免兼容性问题;

主机字节顺序HBO(Host Byte Order)

不同的机器HBO不相同,与CPU的设计有关,数据的顺序是由CPU决定的,而与操作系统无关;
由于这个原因,不同体系结构的机器之间不能直接通信,所以要转换成一种约定的顺序,也就是网络字节顺序,其实就是如同power
pc那样的顺序。在PC开发中有ntohl和htonl函数可以用来进行网络字节和主机字节的转换

假设在x86平台上,有一个int型变量,在内存中的内部由低到高分别是:0x12,0x34,0x56,0x78当通过网络发送该数据时,正确的发送顺序是0x78,0x56,0x34,0x12
X86 系列 CPU都是 little-endian 的,所以int 型变量值为 0x78563412, 网络发送数据时,采用大段,先发送高位再发送低位

2、uint16_t htons(uint16_t hostshort);

htons的功能:将一个无符号短整型数值转换为网络字节序,即大端模式(big-endian)
参数u_short hostshort: 16位无符号整数
返回值:TCP / IP网络字节顺序
htons 是把你机器上的整数转换成“网络字节序”, 网络字节序是 big-endian,也就是整数的高位字节存放在内存的低地址处。 而我们常用的 x86 CPU (intel, AMD) 电脑是 little-endian,也就是整数的低位字节放在内存的低字节处。举个例子吧。假定你的port是0x1234,在网络字节序里 这个port放到内存中就应该显示成addr addr+1,也就是:0x12 0x34;而在x86电脑上,0x1234放到内存中实际是:addr addr+1,也就是:0x34 0x12。htons 的用处就是把实际内存中的整数存放方式调整成“网络字节序”的方式。

第一个问题:为什么使用两个字节,也就是16位来存储。

这个简单一些,因为一个字节只能存储8位2进制数,而计算机的端口数量是65536个,也就是2^16,两个字节。

第二个为题:为什么计算机需要大端模式和小端模式?

小端模式 :强制转换数据不需要调整字节内容,1、2、4字节的存储方式一样。   
大端模式 :符号位的判定固定为第一个字节,容易判断正负。

3、inet_ntoa()

怎样将一个in_addr结构体输出成点数格式?你要用到函数 inet_ntoa()(“ntoa"的含义是"network to ascii”),就像这样:printf(“%s”,inet_ntoa(ina.sin_addr));它将输出IP地址。需要注意的是inet_ntoa()将结构体in_addr作为一个参数,不是长整形。同样需要注意的是它返回的是一个指向一个字符的指针。它是一个由inet_ntoa()控制的静态的固定的指针,所以每次调用 inet_ntoa(),它就将覆盖上次调用时所得的IP地址。

inet_aton(将网络地址转成网络二进制的数字) 相关函数 inet_addr,inet_ntoa 表头文件
#include
#include
#include 定义函数 int inet_aton(const char * cp,struct in_addr *inp); 函数说明
inet_aton()用来将参数cp所指的网络地址字符串转换成网络使用的二进制的数字,然后存于参数inp所指的in_addr结构中。
结构in_addr定义如下 struct in_addr {
unsigned long int s_addr; }; 返回值 成功则返回非0值,失败则返回0。

你可能感兴趣的:(C语言,c语言)