指针运算笔试题解析

题目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;
}
//程序的结果是什么?

(&a+1)为a数组后面的一个元素的地址
*(a+1)==a[1] *(prt-1)==a[4]=5

在这里插入图片描述

题目二:

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

p为结构体类型指针,因此**+1时相当于+20** 0x100014
p被强制类型转换成无符号长整型不是指针因此+1相当于还是+1 0x100001
p被强制类型转换成无符号整型指针因此+1相当于+4 0x100004

在这里插入图片描述

题目三:

#include <stdio.h>
int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	int *p;
	p = a[0];
	printf("%d", p[0]);
	return 0;
}

逗号表达式,因此a[3][2]={1,3,5};
p=a[0]=a[0][0]
因此答案就是1

在这里插入图片描述

题目四:

//假设环境是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;
}

p=a,相当于把a的首元素地址a[0]赋值给了p,也就是下面图中的初始位置
p的类型为整型类型的数组指针,&p[4][2]==&(&(p+4)+2),也就是跳过4*4+2=18个字节到图中蓝色涂鸦的位置。
因此&p[4][2] - &a[4][2]= -4,因为第一个输出时要以%p,也就是地址,因此要将-4转换成补码输出:
10000000 00000000 00000000 00000100
11111111 11111111 11111111 11111011
11111111 11111111 11111111 11111100
也就是 FF FF FF FC
第二个输出是%d,因此直接为-4
在这里插入图片描述
在这里插入图片描述

题目五:

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

(&aa+1)&aa取出整个数组的地址,+1后跳过整个数组
(*(aa+1))aa表示首元素地址,也就是aa[0],+1后就是aa[1]
*(ptr1 -1),因为ptr1为整型指针,-1表示后退一个元素,也就是a[1][4]=10
*(ptr2 -1),因为ptr2为整型指针,-1表示后退一个元素,也就是a[0][4]=5

在这里插入图片描述

题目六:

#include <stdio.h>
int main()
{
	char *a[] = { "work", "at", "alibaba" };
	char**pa = a;
	pa++;
	printf("%s\n", *pa);
	return 0;
}

pa中存放的是w的地址
pa++后pa存放的就是a的地址
因此*pa后的结果就是at

在这里插入图片描述

题目七:

#include <stdio.h>
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;
}
  1. 一开始是这样指向的:
    在这里插入图片描述

  2. ++cpp后:
    在这里插入图片描述
    因此**++cpp的值为:POINT

在这里插入图片描述

*--*++cpp 后指向的就是ENTER,再+3的结果就是ER
4. 在这里插入图片描述
*cpp[-2]==*(*(cpp-2))也就是回到cpp[0],最终指向FIRST,再+3后就是ST
5.
在这里插入图片描述

cpp[-1][-1]==*(*(cpp-1)-1)因为上个代码并没有改变cpp的地址,因此一开始cpp指向的还是cpp+2,在-1后指向cpp-1,就是cp[c+2],-1后就是cp[c+1],也就是NEW,+1后就是EW
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值