C/C++测试代码
#include <iostream>
int main(int argc, char* argv[])
{
int n = 0x11223344;
int i = 0;
for( i = 0; i < 4; i++)
{
printf("%02x", *(((char*)&n)+i));
}
printf("\n");
return 0;
}
Centos 5.3 + 4.1.2
编译命令:g++ ytest.cpp -o test.ex
44 33 22 11
WinXP + VS2003测试结果
编译命令:cl ytest.cpp
44 33 22 11
Java测试代码
要特别注意的是,移位操作并不是内存移位,而是数字的字面移位,如下代码无论在何种操作系统,何种语言,结果将是一样
所以以下代码只适用于Java
int x = 0x11223344;
System.out.printf("%02x ", 0xFF&(x>>24));
System.out.printf("%02x ", 0xFF&(x>>16));
System.out.printf("%02x ", 0xFF&(x>>8));
System.out.printf("%02x", 0xFF&x);
不同的CPU有不同的字节序类型,这些字节序是指整数在内存中保存的顺序。
最常见的有两种:
1. Little-endian:将低序字节存储在起始地址(低位编址)
2. Big-endian:将高序字节存储在起始地址(高位编址)
LE(little-endian):
最符合人的思维的字节序
地址低位存储值的低位
地址高位存储值的高位
怎么讲是最符合人的思维的字节序,是因为从人的第一观感来说
低位值小,就应该放在内存地址小的地方,也即内存地址低位
反之,高位值就应该放在内存地址大的地方,也即内存地址高位
BE(big-endian):
最直观的字节序
地址低位存储值的高位
地址高位存储值的低位
为什么说直观,不要考虑对应关系
只需要把内存地址从左到右按照由低到高的顺序写出
把值按照通常的高位到低位的顺序写出
两者对照,一个字节一个字节的填充进去
例子1:在内存中双字0x01020304(DWORD)的存储方式。
内存地址 4000 4001 4002 4003
LE 04 03 02 01
BE 01 02 03 04
注:每个地址存1个字节,每个字有4个字节。2位16进制数是1个字节(0xFF=11111111)。
例子2:如果我们将0x1234abcd写入到以0x0000开始的内存中,则结果为
big-endian little-endian
0x0000 0x12 0xcd
0x0001 0x23 0xab
0x0002 0xab 0x34
0x0003 0xcd 0x12
x86系列的CPU都是little-endian的字节序。