最常見的有兩種
1. Little endian:將低序字節(jié)存儲(chǔ)在起始地址
2. Big endian:將高序字節(jié)存儲(chǔ)在起始地址
LE little-endian
最符合人的思維的字節(jié)序
地址低位存儲(chǔ)值的低位
地址高位存儲(chǔ)值的高位
怎么講是最符合人的思維的字節(jié)序,是因?yàn)閺娜说牡谝挥^感來說
低位值小,就應(yīng)該放在內(nèi)存地址小的地方,也即內(nèi)存地址低位
反之,高位值就應(yīng)該放在內(nèi)存地址大的地方,也即內(nèi)存地址高位
BE big-endian
最直觀的字節(jié)序
地址低位存儲(chǔ)值的高位
地址高位存儲(chǔ)值的低位
為什么說直觀,不要考慮對(duì)應(yīng)關(guān)系
只需要把內(nèi)存地址從左到右按照由低到高的順序?qū)懗?nbsp;
把值按照通常的高位到低位的順序?qū)懗?nbsp;
兩者對(duì)照,一個(gè)字節(jié)一個(gè)字節(jié)的填充進(jìn)去
例子:在內(nèi)存中雙字0x01020304(DWORD)的存儲(chǔ)方式
內(nèi)存地址
4000 4001 4002 4003
LE 04 03 02 01
BE 01 02 03 04
例子:如果我們將0x1234abcd寫入到以0x0000開始的內(nèi)存中,則結(jié)果為
big-endian little-endian
0x0000 0x12 0xcd
0x0001 0x23 0xab
0x0002 0xab 0x34
0x0003 0xcd 0x12
x86系列CPU都是little-endian的字節(jié)序.
網(wǎng)絡(luò)字節(jié)順序是TCP/IP中規(guī)定好的一種數(shù)據(jù)表示格式,它與具體的CPU類型、操作系統(tǒng)等無關(guān),從而可以保證數(shù)據(jù)在不同主機(jī)之間傳輸時(shí)能夠被正確解釋。網(wǎng)絡(luò)字節(jié)順序采用big endian排序方式。
為了進(jìn)行轉(zhuǎn)換 bsd socket提供了轉(zhuǎn)換的函數(shù) 有下面四個(gè)
htons 把unsigned short類型從主機(jī)序轉(zhuǎn)換到網(wǎng)絡(luò)序
htonl 把unsigned long類型從主機(jī)序轉(zhuǎn)換到網(wǎng)絡(luò)序
ntohs 把unsigned short類型從網(wǎng)絡(luò)序轉(zhuǎn)換到主機(jī)序
ntohl 把unsigned long類型從網(wǎng)絡(luò)序轉(zhuǎn)換到主機(jī)序
在使用little endian的系統(tǒng)中 這些函數(shù)會(huì)把字節(jié)序進(jìn)行轉(zhuǎn)換
在使用big endian類型的系統(tǒng)中 這些函數(shù)會(huì)定義成空宏
同樣 在網(wǎng)絡(luò)程序開發(fā)時(shí) 或是跨平臺(tái)開發(fā)時(shí) 也應(yīng)該注意保證只用一種字節(jié)序 不然兩方的解釋不一樣就會(huì)產(chǎn)生bug.
注:
1、網(wǎng)絡(luò)與主機(jī)字節(jié)轉(zhuǎn)換函數(shù):htons ntohs htonl ntohl (s 就是short l是long h是host n是network)
2、不同的CPU上運(yùn)行不同的操作系統(tǒng),字節(jié)序也是不同的,參見下表。
處理器 操作系統(tǒng) 字節(jié)排序
Alpha 全部 Little endian
HP-PA NT Little endian
HP-PA UNIX Big endian
Intelx86 全部 Little endian <-----x86系統(tǒng)是小端字節(jié)序系統(tǒng)
Motorola680x() 全部 Big endian
MIPS NT Little endian
MIPS UNIX Big endian
PowerPC NT Little endian
PowerPC 非NT Big endian <-----PPC系統(tǒng)是大端字節(jié)序系統(tǒng)
RS/6000 UNIX Big endian
SPARC UNIX Big endian
IXP1200 ARM核心 全部 Little endian
下面是一個(gè)檢驗(yàn)本機(jī)字節(jié)序的簡(jiǎn)便方法:
//判斷本機(jī)的字節(jié)序
//返回true表為小段序。返回false表示為大段序
bool am_little_endian ()
{
unsigned short i=1;
return (int)*((char *)(&i)) ? true : false;
}
int main()
{
if(am_little_endian())
{
printf("本機(jī)字節(jié)序?yàn)樾《涡?\n");
}
else
{
printf("本機(jī)字節(jié)序?yàn)榇蠖涡?\n");
}
return 0;
}