有关【指针运算】的经典笔试题

题目1:

#include <stdio.h>

int main()
{
	int a[5] = { 1, 2, 3, 4, 5 };
	int* ptr = (int*)(&a + 1);
	printf("%d,%d", *(a + 1), *(ptr - 1));

	return 0;
}
//程序的输出结果是什么?  2  5

解析:
代码的内存布局如图:
在这里插入图片描述
- 考点:

  1. 理解数组名与&数组名的区别。
    数组名一般是指数组首元素地址(除sizeof(数组名)外);&数组名取出的是整个数组的地址。两者的值相同,意义不一样。
  2. 理解指针±整数操作。
    指针±整数的步幅是多少,取决于指针的类型,如整型指针一次解引用访问4个字节,字符指针一次解引用访问1个字节。

题目2:

#include <stdio.h>

//在X86环境下 
//假设结构体的⼤⼩是20个字节 
//程序输出的结果是啥? 

struct Test
{
	int Num;
	char* pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}* p  = (struct Test*)0x100000;

int main()
{
	printf("%p\n", p + 0x1);
	printf("%p\n", (unsigned long)p + 0x1);
	printf("%p\n", (unsigned int*)p + 0x1);

	return 0;
}

解析:
代码解析如图:![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/3d19e61083634670b41a510572e36c5f.png
所以,第一个printf语句中,p+0x1就等价于指针±整数操作,步幅取决于指针类型的大小,由题可知结构体大小为20个字节,所以结果为0x100000+14(20要转换为十六进制)=0x100014。
第二个printf语句中,p是无符号整型,就是整数,整数+1,结果就是0x100001。
第三个printf语句中,p是整型指针,+1跳过4个字节,结果为0x100004。

- 考点:指针±整数。指针±整数的步幅是多少,取决于指针的类型。

题目3:

#include <stdio.h>

int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	
	int* p;
	p = a[0];
	printf("%d", p[0]);  //1
	
	return 0;
}

解析:
代码内存布局如图:
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/03bb0fb918f248f29a8f6f72279d5ece.png
这道题也是考查对数组名的理解,a[0]是二维数组第一行的数组名,而数组名又是数组首元素地址,所以p[0]=1。
- 考点:对二维数组的数组名的理解。

题目4:

//假设环境是x86环境,程序输出的结果是啥? 

#include <stdio.h>

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;
}

解析:
代码的解析和内存布局如图:
在这里插入图片描述
|指针-指针| = 两者之间元素的个数。由于数组在内存中是由低地址向高地址存储的,由图可知,两个地址的差是-4,%d打印有符号整型,结果为-4;但是%p是以十六进制打印补码,-4的原码为10000000000000000000000000000100,所以补码为11111111111111111111111111111100,转换为十六进制为FFFFFFFC。

题目5:

#include <stdio.h>

int main()
{
 int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
 
 int *ptr1 = (int *)(&aa + 1);
 int *ptr2 = (int *)(*(aa + 1));
 
 printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));
 
 return 0;
}

解析:
代码内存布局如图:
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/03371283b6794582ad51396e40e1a4fd.png
&aa取出的是整个二维数组的地址,+1跳过整个二维数组;而aa要理解为二维数组第一行的地址,+1跳过一整行。

题目6:

#include <stdio.h>

int main()
{
 char *a[] = {"work","at","alibaba"};
 
 char**pa = a;
 pa++;
 printf("%s\n", *pa);// at
 
 return 0;
}

解析:
代码的内存布局如图:
在这里插入图片描述
这题主要考察对指针数组的理解和二级指针的理解。
指针数组是数组,里面的元素是地址,二级指针存放的是一级指针的地址。

题目7:

int main()
{
 char *c[] = {"ENTER","NEW","POINT","FIRST"};
 char**cp[] = {c+3,c+2,c+1,c};
 char***cpp = cp;
 
 printf("%s\n", **++cpp);
 printf("%s\n", *--*++cpp+3);
 printf("%s\n", *cpp[-2]+3);
 printf("%s\n", cpp[-1][-1]+1);
 
 return 0;
}

解析:
代码的内存布局如图:
在这里插入图片描述
这个题较为复杂,一定要正确的画出内存布局。并且一定要注意"++cpp和–cpp"这种操作是有"副作用"的运算,它会改变本身的内容!

  • 29
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 20
    评论
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值