作者:@匿名者Unit
目录
- 指针笔试题
- eg1.
- eg2.
- eg3.
- eg4.
指针笔试题
eg1.
我们先来看第一题:
int main()
{
int a[4] = { 1, 2, 3, 4 };
int *ptr1 = (int *)(&a + 1);
int *ptr2 = (int *)((int)a + 1);
printf( "%x,%x", ptr1[-1], *ptr2);
return 0;
}
我们先来分析ptr1,首先 &a 是取出了整个数组的地址接着+1越界到元素4后面的元素个数为4的数组 类型为int(*)[4],所以 ptr[-1] 是元素4 .
ptr2的分析较难些:首先a是数组首元素的地址,将地址强制转换为整形,我们假设a首元素的地址是0x0000ff40整形加一就变成了0x0000ff41,说明ptr2所存储的地址向前走了一字节,那它代表多大的数呢。在我的电脑上VS2019-x86是小端存储方式,我们将数组在内存中的存储画出来。
结果如下:
eg2.
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int *p;
p = a[0];
printf( "%d", p[0]);
return 0;
}
这题有个大坑,一个不注意就以为答案是0.其实a在初始化时花括号中的是逗号表达式,并不是初始化列表。所以a中的元素其实是 {1,3}{5,0}{0,0},所以 p[0] 是1
eg3.
int main()
{
int a[5][5];
int(*p)[4];
p = a;
printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
return 0;
}
p是一个数组指针指向大小为4的整形数组。
又因为指针减指针代表中间的元素个数,结果就是-4,但是用 %p格式打印时是无符号的 打印的是-4的补码,所以是一个很大的数
eg4.
struct Test
{
int Num;
char* pcName;
short sDate;
char cha[2];
short sBa[4];
}* p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()
{
p = (struct Test*)0x100000;
printf("%p\n", p + 0x1);
printf("%p\n", (unsigned long)p + 0x1);
printf("%p\n", (unsigned int*)p + 0x1);
return 0;
}
这题主要是知道指针类型决定步长,struct test类型加一等于p加20,即0x00100014
p强转为无符号长整型加1即是直接加一为0x00100001,unsigned int类型加一为4个字节,即为 0x00100004 总之就是类型决定了步长。