网络编程中的字节序函数htonl()、htons()、ntohl()和ntohs()
目录
1,网络字节序和主机字节序
2. 函数的具体作用
2.1,htonl(Host to Network Long)
2.2,htons(Host to Network Short)
2.3,ntohl(Network to Host Long)
2.4,ntohs(Network to Host Short)
3,为什么需要这些函数?
4,适用范围
5,如何判断当前平台的字节序?
6,常见架构的字节序
1,网络字节序和主机字节序
-
网络字节序:
规定采用 大端字节序(Big Endian),即 高位字节存储在低地址,这是网络协议(如 TCP/IP)使用的标准。 -
主机字节序:
是指主机(计算机)的 CPU 如何存储多字节数据:- 大端字节序:高位字节存储在低地址。
- 小端字节序:低位字节存储在低地址(如 x86、x86_64)。
不同平台的字节序可能不同,因此在网络通信中必须统一使用网络字节序(大端)。
2. 函数的具体作用
2.1,htonl
(Host to Network Long)
- 作用:将 32 位整数 从主机字节序转换为网络字节序(大端)。
- 函数原型:
uint32_t htonl(uint32_t hostlong);
- 示例:
uint32_t hostValue = 0x12345678; // 在小端机器上,内存中为 0x78 0x56 0x34 0x12
uint32_t networkValue = htonl(hostValue); // 转换为网络字节序:0x12 0x34 0x56 0x78
2.2,htons
(Host to Network Short)
- 作用:将 16 位整数 从主机字节序转换为网络字节序(大端)。
- 函数原型:
uint16_t htons(uint16_t hostshort);
- 示例:
uint16_t hostValue = 0x1234; // 在小端机器上,内存中为 0x34 0x12
uint16_t networkValue = htons(hostValue); // 转换为网络字节序:0x12 0x34
2.3,ntohl
(Network to Host Long)
- 作用:将 32 位整数 从网络字节序转换为主机字节序。
- 函数原型:
uint32_t ntohl(uint32_t netlong);
- 示例:
uint32_t networkValue = 0x78563412; // 假设收到的网络数据
uint32_t hostValue = ntohl(networkValue); // 转换为主机字节序(在小端机器上):0x12345678
2.4,ntohs
(Network to Host Short)
- 作用:将 16 位整数 从网络字节序转换为主机字节序。
- 函数原型:
uint16_t ntohs(uint16_t netshort);
- 示例:
uint16_t networkValue = 0x3412; // 假设收到的网络数据
uint16_t hostValue = ntohs(networkValue); // 转换为主机字节序(在小端机器上):0x1234
3,为什么需要这些函数?
在网络通信中,主机之间可能使用不同的字节序。因此,发送方需要将数据转换为网络字节序,接收方需要将网络字节序转换回主机字节序。这些函数可以确保数据在网络上传输时的字节序是 一致的,从而避免因字节序不同导致的数据错误。
例如:
假设两个主机进行 UDP 通信,传输一个 32 位整数:
发送端(小端架构):
uint32_t data = 0x12345678;
uint32_t networkData = htonl(data); // 转换为大端
sendto(sockfd, &networkData, sizeof(networkData), 0, ...);
接收端(小端架构):
uint32_t receivedData;
recvfrom(sockfd, &receivedData, sizeof(receivedData), 0, ...);
uint32_t hostData = ntohl(receivedData); // 转换回小端
这样,无论发送端和接收端的架构如何,数据都可以正确传输和解析。
4,适用范围
函数 | 适用数据类型 | 转换方向 |
---|---|---|
htonl | 32 位整数 | 主机字节序 → 网络字节序 |
htons | 16 位整数 | 主机字节序 → 网络字节序 |
ntohl | 32 位整数 | 网络字节序 → 主机字节序 |
ntohs | 16 位整数 | 网络字节序 → 主机字节序 |
5,如何判断当前平台的字节序?
#include <iostream>
int main() {
uint32_t num = 1;
if (*(reinterpret_cast<uint8_t*>(&num)) == 1) {
std::cout << "小端字节序" << std::endl;
} else {
std::cout << "大端字节序" << std::endl;
}
return 0;
}
- 如果输出 小端字节序,则当前主机使用 小端。
- 如果输出 大端字节序,则当前主机使用 大端。
6,常见架构的字节序
处理器架构 | 字节序 |
---|---|
x86、x86_64 | 小端 |
ARM(大多数情况) | 小端 |
PowerPC | 大端 |
SPARC | 大端 |