思路:联合体union的存放顺序是所有成员都从低地址开始存放,利用该特性就可以轻松地获得了
CPU对内存采用Little-endian还是Big-endian模式读写。
给出c代码,linux其实也差不了多少
#include
using namespace std;
union Judge
{
short s;
char ch[sizeof(short)];
};
Judge flag;
int main()
{
flag.s = 0x0102;
if ( sizeof(short) !=2 )
{
cout
int main(void)
{
int i = 1;
unsigned char *pointer;
pointer = (unsigned char *)&i;
if(*pointer)
{
printf("litttle_endian");
}
else
{
printf("big endian\\n");
}
return 0;
}
C中的数据类型都是从内存的低地址向高地址扩展,取址运算"&"都是取低地址。小端方式中(i占至少两个字节的长度)则i所分配的内存最小地址那个字节中就存着1,其他字节是0。大端的话则1在i的最高地址字节处存放,char是一个字节,所以强制将char型量p指向i,则p指向的一定是i的最低地址,那么就可以判断p中的值是不是1来确定是不是小端。
方法2:
#include
int main(void)
{
union {
short a;
char ch;
} u;
u.a = 1;
if (u.ch == 1)
{
printf("Littel endian\\n");
}
else
{
printf("Big endian\\n");
}
}
利用联合体的特点,数据成员共享内存空间,union中元素的起始地址都是相同的——位于联合的开始。 用char来截取感兴趣的字节。
四、需要考虑大小端(字节顺序)的情况
1、所写的程序需要向不同的硬件平台迁移,说不定哪一个平台是大端还是小端,为了保证可移植性,一定提前考虑好。
2. 在不同类型的机器之间通过网络传送二进制数据时。 一个常见的问题是当小端法机器产生的数据被发送到大端法机器或者反之时,接受程序会发现,字(word)里的字节(byte)成了反序的。为了避免这类问 题,网络应用程序的代码编写必须遵守已建立的关于字节顺序的规则,以确保发送方机器将它的内部表示转换成网络标准,而接受方机器则将网络标准转换为它的内部标准。
3. 当阅读表示整数的字节序列时。这通常发生在检查机器级程序时,e.g.:反汇编得到的一条指令:
80483bd: 01 05 64 94 04 08 add �x, 0x8049464
3. 当编写强转的类型系统的程序时。
例1:如写入的数据为u32型,但是读取的时候却是char型的:
如:0x1234, 大端读取为12时,小端独到的是34。
例2:将buffer中的数转换为整型:
如:int src = 0x1234
memcpy(buffer, src, sizeof(int));
大端:buffer[4] = {0x01, 0x02, 0x03, 0x04};
小端:buffer[4] = {0x04, 0x03, 0x02, 0x01};
六、提高程序的可移植性
使用宏编译
#ifdef LITTLE_ENDIAN
//小端的代码
#else
//大端的代码
#endif
七、大、小端之间的转换
1、小端转换为大端
#include
void show_byte(char *addr, int len)
{
int i;
for (i = 0; i >= 8;
}
return result;
}
int main(void)
{
int i;
int ret;
i = 0x1234567;
show_byte((char *)&i, sizeof(int));
ret = endian_convert(i);
show_byte((char *)&ret, sizeof(int));
return 0;
}
本文来自:我爱研发网(52RD.com) - R&D大本营
详细出处:http://www.52rd.com/bbs/Dispbbs.asp?BoardID=118&ID=211191