网络数据的大小端问题
不同 CPU 中,4 字節(jié)整數(shù) 1 在內(nèi)存空間的存儲(chǔ)方式是不同的。4 字節(jié)整數(shù) 1 可用 2 進(jìn)制表示如下:
00000000 00000000 00000000 00000001
有些 CPU 以上面的順序存儲(chǔ)到內(nèi)存,另外一些 CPU 則以倒序存儲(chǔ),如下所示:
00000001 00000000 00000000 00000000
若不考慮這些就收發(fā)數(shù)據(jù)會(huì)發(fā)生問(wèn)題,因?yàn)楸4骓樞虻牟煌馕吨鴮?duì)接收數(shù)據(jù)的解析順序也不同。
大端序和小端序
CPU 向內(nèi)存保存數(shù)據(jù)的方式有兩種:
- 大端序(Big Endian):高位字節(jié)存放到低位地址(高位字節(jié)在前)。
- 小端序(Little Endian):高位字節(jié)存放到高位地址(低位字節(jié)在前)。
僅憑描述很難解釋清楚,不妨來(lái)看一個(gè)實(shí)例。假設(shè)在 0x20 號(hào)開(kāi)始的地址中保存 4 字節(jié) int 型數(shù)據(jù) 0x12345678,大端序 CPU 保存方式如下圖所示:
圖1:整數(shù) 0x12345678 的大端序字節(jié)表示
對(duì)于大端序,最高位字節(jié) 0x12 存放到低位地址,最低位字節(jié) 0x78 存放到高位地址。小端序的保存方式如下圖所示:
圖2:整數(shù) 0x12345678 的小端序字節(jié)表示
不同 CPU 保存和解析數(shù)據(jù)的方式不同(主流的 Intel 系列 CPU 為小端序),小端序系統(tǒng)和大端序系統(tǒng)通信時(shí)會(huì)發(fā)生數(shù)據(jù)解析錯(cuò)誤。因此在發(fā)送數(shù)據(jù)前,要將數(shù)據(jù)轉(zhuǎn)換為統(tǒng)一的格式——網(wǎng)絡(luò)字節(jié)序(Network Byte Order)。網(wǎng)絡(luò)字節(jié)序統(tǒng)一為大端序。
主機(jī) A 先把數(shù)據(jù)轉(zhuǎn)換成大端序再進(jìn)行網(wǎng)絡(luò)傳輸,主機(jī) B 收到數(shù)據(jù)后先轉(zhuǎn)換為自己的格式再解析。
網(wǎng)絡(luò)字節(jié)序轉(zhuǎn)換函數(shù)
在《bind()和connect()函數(shù):綁定套接字并建立連接》一節(jié)中講解了 sockaddr_in 結(jié)構(gòu)體,其中就用到了網(wǎng)絡(luò)字節(jié)序轉(zhuǎn)換函數(shù),如下所示:
htons() 用來(lái)將當(dāng)前主機(jī)字節(jié)序轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序,其中h代表主機(jī)(host)字節(jié)序,n代表網(wǎng)絡(luò)(network)字節(jié)序,s代表short,htons 是 h、to、n、s 的組合,可以理解為”將 short 型數(shù)據(jù)從當(dāng)前主機(jī)字節(jié)序轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序“。
常見(jiàn)的網(wǎng)絡(luò)字節(jié)轉(zhuǎn)換函數(shù)有:
- htons():host to network short,將 short 類型數(shù)據(jù)從主機(jī)字節(jié)序轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序。
- ntohs():network to host short,將 short 類型數(shù)據(jù)從網(wǎng)絡(luò)字節(jié)序轉(zhuǎn)換為主機(jī)字節(jié)序。
- htonl():host to network long,將 long 類型數(shù)據(jù)從主機(jī)字節(jié)序轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序。
- ntohl():network to host long,將 long 類型數(shù)據(jù)從網(wǎng)絡(luò)字節(jié)序轉(zhuǎn)換為主機(jī)字節(jié)序。
通常,以s為后綴的函數(shù)中,s代表 2 個(gè)字節(jié) short,因此用于端口號(hào)轉(zhuǎn)換;以l為后綴的函數(shù)中,l代表 4 個(gè)字節(jié)的 long,因此用于 IP 地址轉(zhuǎn)換。
舉例說(shuō)明上述函數(shù)的調(diào)用過(guò)程:
運(yùn)行結(jié)果:
Host ordered port: 0x1234
Network ordered port: 0x3412
Host ordered address: 0x12345678
Network ordered address: 0x78563412
另外需要說(shuō)明的是,sockaddr_in 中保存 IP 地址的成員為 32 位整數(shù),而我們熟悉的是點(diǎn)分十進(jìn)制表示法,例如 127.0.0.1,它是一個(gè)字符串,因此為了分配 IP 地址,需要將字符串轉(zhuǎn)換為 4 字節(jié)整數(shù)。
inet_addr() 函數(shù)可以完成這種轉(zhuǎn)換。inet_addr() 除了將字符串轉(zhuǎn)換為 32 位整數(shù),同時(shí)還進(jìn)行網(wǎng)絡(luò)字節(jié)序轉(zhuǎn)換。請(qǐng)看下面的代碼:
運(yùn)行結(jié)果:
Network ordered integer addr: 0x4030201
Error occured!
從運(yùn)行結(jié)果可以看出,inet_addr() 不僅可以把 IP 地址轉(zhuǎn)換為 32 位整數(shù),還可以檢測(cè)無(wú)效 IP 地址。
注意:為 sockaddr_in 成員賦值時(shí)需要顯式地將主機(jī)字節(jié)序轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序,而通過(guò) write()/send() 發(fā)送數(shù)據(jù)時(shí) TCP 協(xié)議會(huì)自動(dòng)轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序,不需要再調(diào)用相應(yīng)的函數(shù)。
總結(jié)
以上是生活随笔為你收集整理的网络数据的大小端问题的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: mpi由于目标计算机积极拒绝,windo
- 下一篇: gethostbyname()函数:通过