大小端模式对 union 类型数据的影响

下面再看一个例子:
union
{
int i;
char a[2];
}*p, u;
p = &u;
p->a[0] = 0x39;
p->a[1] = 0x38;
p.i 的值应该为多少呢?
这里需要考虑存储模式:大端模式和小端模式。
大端模式( Big_endian ):字数据的 高字节 存储在 低地址 中,而字数据的 低字节 则存放
高地址 中。
小端模式( Little_endian ):字数据的 高字节 存储在 高地址 中,而字数据的 低字节 则存放
低地址 中。
union 型数据所占的空间等于其最大的成员所占的空间。对 union 型的成员的存取都是
相对于该联合体基地址的偏移量为 0 处开始,也就是联合体的访问不论对哪个变量的存取都
是从 union 的首地址位置开始。如此一解释,上面的问题是否已经有了答案呢?
1.15.2,如何用程序确认当前系统的存储模式?
上述问题似乎还比较简单,那来个有技术含量的:请写一个 C 函数,若处理器是
Big_endian 的,则返回 0 ;若是 Little_endian 的,则返回 1
先分析一下,按照上面关于大小端模式的定义,假设 int 类型变量 i 被初始化为 1
以大端模式存储,其内存布局如下图:
以小端模式存储,其内存布局如下图: 变量 i 4 个字节,但只有一个字节的值为 1 ,另外三个字节的值都为 0 。如果取出低
地址上的值为 0 ,毫无疑问,这是大端模式;如果取出低地址上的值为 1 ,毫无疑问,这是
小端模式。既然如此,我们完全可以利用 union 类型数据的特点:所有成员的起始地址一致。
到现在,应该知道怎么写了吧?参考答案如下:
int checkSystem( )
{
union check
{
int i;
char ch;
} c;
c.i = 1;
return (c.ch ==1);
}
现在你可以用这个函数来测试你当前系统的存储模式了。当然你也可以不用函数而直
接去查看内存来确定当前系统的存储模式。如下图:
图中 0x01 的值存在低地址上,说明当前系统为小端模式。
不过要说明的一点是,某些系统可能同时支持这两种存储模式,你可以用硬件跳线或
在编译器的选项中设置其存储模式。
留个问题:
x86 系统下,输出的值为多少?
#include
int main()
{
int a[5]={1,2,3,4,5};
int *ptr1=(int *)(&a+1); int *ptr2=(int *)((int)a+1);
printf("%x,%x",ptr1[-1],*ptr2);
return 0;
}

你可能感兴趣的:(c语言)