记录一次htonl和ntohl的使用方法和差别

实例:

在研究这两个函数时,总是遇到htonl和ntohl输出的值是一样的情况,明明一个是主机序转网络序,一个是网络序转主机序,为什么值会输出一样呢?请看代码:

	uint32_t ulTemp = 1234;  //16进制为00 00 04D2(大端序)  ,
实质我的机器(小端机)储存为:D2 04 00 00四字节
	ulTemp = htonl(ulTemp); //机器为主机序,使用htonl转为大端序后,
ulTemp内存为: 00 00 04 D2

	uint8_t mysz[4] = {0xff};
	memcpy((void *)mysz, (void *)&ulTemp, 4);
	printf("%02x%02x%02x%02x\n",mysz[0],mysz[1],mysz[2],mysz[3]);	
//结果为:00 00 04D2 ,这里看起来没问题,是正常的。

好,上面看起来符合正常逻辑,没问题。

接下来看下面使用ntohl来看看情况,个人认为这是比较诡异的部分:


uint32_t ulTemp = 1234;  //16进制为00 00 04D2(大端序)  ,
实质我的机器(小端机)储存为:D2 04 00 00四字节
	ulTemp = ntohl(ulTemp); //使用该函数后,
ulTemp在机器内存中存储实质变为:00 00 04 D2 ??
问题:为什么我调用了ntohl转为小端序了,ulTemp 怎么内存结构是大端序的样子呢?

	uint8_t mysz[4] = {0xff};
	memcpy((void *)mysz, (void *)&ulTemp, 4);
	printf("%02x%02x%02x%02x\n",mysz[0],mysz[1],mysz[2],mysz[3]);
 //结果:00 00 04 D2

看完上面代码你会发现,使用ntohl并没有转为你想要的小端序结果(D2 04 00 00 ),依然还是大端结果00 00 04 D2,和前面的例子输出一样,为什么呢?

个人认为:在调用ntohl时,其实是先将ulTemp转为大端序(00 00 04 D2),然后再看机器的字节序是大端还是小端,如果是小端的话,就不再转了(00 00 04 D2)。如果是大端,就还需要再转一次(D2 04 00 00)。

在调用htonl时,先将ulTemp转为小端序(D2 04 00 00),然后再看机器字节序是大端还是小端,如果是小端的话,就还需要再转为大端(00 00 04 D2)。如果是大端,就不需要再转了

个人结论,不一定准确:判断ntohl和htonl的值时,我们先判断第一个字母,如果是n,说明该变量要转为大端序,再看机器的字节序是否为小端序,是的话就不转序,不是就再转一道序。如果第一个字母是h,先将该变量转为小端序,再看机器字节序是否为大端序,是的话就不转序,不是就再转一次序。这样,无论是使用htonl还是使用ntohl,其最终在内存中都是一样的,这样就解释的通了。

这几个函数都是用来进行网络字节序和主机字节序之间的转换的。 - `htonl` (Host TO Network Long):将一个 32 位的主机字节序整数转换为网络字节序整数。如果主机字节序是大端字节序,则不做任何改变;如果主机字节序是小端字节序,则将其转换为大端字节序。返回网络字节序整数。 ```c++ #include <arpa/inet.h> uint32_t htonl(uint32_t hostlong); ``` - `htons` (Host TO Network Short):将一个 16 位的主机字节序整数转换为网络字节序整数。如果主机字节序是大端字节序,则不做任何改变;如果主机字节序是小端字节序,则将其转换为大端字节序。返回网络字节序整数。 ```c++ #include <arpa/inet.h> uint16_t htons(uint16_t hostshort); ``` - `ntohl` (Network TO Host Long):将一个 32 位的网络字节序整数转换为主机字节序整数。如果当前主机字节序是大端字节序,则不做任何改变;如果当前主机字节序是小端字节序,则将其转换为小端字节序。返回主机字节序整数。 ```c++ #include <arpa/inet.h> uint32_t ntohl(uint32_t netlong); ``` - `ntohs` (Network TO Host Short):将一个 16 位的网络字节序整数转换为主机字节序整数。如果当前主机字节序是大端字节序,则不做任何改变;如果当前主机字节序是小端字节序,则将其转换为小端字节序。返回主机字节序整数。 ```c++ #include <arpa/inet.h> uint16_t ntohs(uint16_t netshort); ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值