1. 指针问题
int main()
{
int a[5] = {1, 2, 3, 4, 5};
int *ptr=(int *)(&a + 1);
printf("%d, %d",*(a + 1), *(ptr -1 ));
return 0;
}
&a+1是将整个数组a的长度再偏离一个int,赋给指针ptr,所以*ptr指向的是a[5],*(ptr -1 )是a[4]=5。*(a + 1)表示的就是a[1]。
2. 结构体长度(32位机)
struct MyStruct
{
int i;
char c;
struct InnerStruct
{
int i;
long l;
double d;
char c;
} innerStruct;
};
union MyUnion
{
int i;
char c;
};
int main()
{
printf("%d, %d", sizeof(MyStruct), sizeof(MyUnion));
}
结构体长度和结构体内部定义的变量类型还有先后顺序有关。
上述结构体,先定义的int为4个字节,char为一个字节,所以第二个char补齐为4个字节。内部结构体int 为4个字节,32位机long为4个字节,double为8字节,char在double之后所以补齐为8个字节。加起来4+4+4+4+8+8=32。
偏移规则:
(1)每个成员的偏移量都必须是当前成员所占内存大小的整数倍如果不是编译器会在成员之间加上填充字节。
(2)当所有成员大小计算完毕后,编译器判断当前结构体大小是否是结构体中最宽的成员变量大小的整数倍 如果不是会在最后一个成员后做字节填充。要么是4的倍数,要么是8的倍数。
3. 联合体的长度
联合体所占的内存长度等于其最长成员的长度,也有叫做共用体
1)大小足够容纳最宽的成员;
2)大小能保证被其包含的基本数据类型的最大的长度所整除。