【指针笔试题】


前言

前段时间收藏了一些指针笔试题,简单和困难模式都有。打算暑假来做一做,却拖了很久,在收藏夹吃灰了,最近我夜观天象,今日又晴空万里,伸出手指一算,发现,今天是个做题的好天气,我连忙翻出了被遗忘在收藏夹里的练习题,今日就来攻破它!


一、指针笔试题

笔试题一

代码如下:

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

解析:

int* ptr = (int*)(&a + 1);	

&a表示取出整个数组的地址,&a+1表示跳过整个数组,ptr在数组末端后面的地址。

printf("%d %d", *(a + 1),*(ptr-1));

a表示数组首元素地址,a+1表示数组第二个元素地址,(a+1)表示数组第二个元素,也就是2,
ptr-1表示向一个元素的地址,就是数组最后一个元素的地址,就是5的地址,
(ptr-1)就是5

画图解析:
在这里插入图片描述

笔试题二

代码如下

//由于还没学习结构体,这里告知结构体的大小是20个字节
struct Test
{
	int Num;
	char* pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}* p = (struct Test*)0x100000;

//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()
{
	printf("%p\n", p + 0x1);
	printf("%p\n", (unsigned int)p + 0x1);
	printf("%p\n", (unsigned int*)p + 0x1);
	
	return 0;
}

解析

printf("%p\n", p + 0x1);	

p是一个结构体指针,+1跳过一个结构体
已知p结构体大小为20个字节,所以0x100000+20=0x100014,注意是转换成16进制。

printf("%p\n", (unsigned int)p + 0x1);

p的结构体类型被强制类型转换成了无符号长整型,已经不是地址了,变成了整数,整数+1就是+1,跟1+1=2一样的道理,所以0x100000+1=0x100001

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

p被强制类型转换成无符号整型指针,跟结构体指针的方法一样,+1跳过4个字节。0x100000+4=0x100004

笔试题三

int main()
{
	int a[4] = { 1, 2, 3, 4 };
	int* ptr1 = (int*)(&a + 1);//4后面
	int* ptr2 = (int*)((int)a + 1);//
	printf("%x,%x", ptr1[-1], *ptr2);
	return 0;
}

解析:

int* ptr1 = (int*)(&a + 1);

&a+1跳过整个数组

int* ptr2 = (int*)((int)a + 1);

a本来是首元素地址,a的类型是int*,但是(int)a意思是a被强制类型转换成整型了,那么(int)+1就不是跳过一个元素,而是整数+1

printf("%x,%x", ptr1[-1], *ptr2);

ptr1[-1]=ptr1+(-1)=*(ptr1-1),ptr1-1向前访问一个元素地址,解引用得到4

画图解析:
在这里插入图片描述

笔试题四

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

解析:

int a[3][2] = { (0, 1), (2, 3), (4, 5) };

逗号表达式:从左向右计算,整个表达式的结果是最后一个表达式的结果
注意:这里是三个逗号表达式,一个括号里算出一个最大值
所以实际上是int a[3][2]={1,3,5};

	int* p;
	p = a[0];

a[0]是二维数组第一行的数组名,a[0]没有单独放在sizeof内部,也没有&修饰,所以是第一行的首元素地址,p是一个整型指针,存放a[0]的地址

printf("%d", p[0]);

p[0]=*(p+0)=*p
*p解引用的是存放在p里面的地址所指向的元素,就是1

画图解析:
在这里插入图片描述

笔试题五

int main()
{
	int a[5][5];//五列五行
	int(*p)[4]; //*p是一个数组指针,存放数组地址的指针,4行4列
	p = a;//把a赋给p
	printf("%p, %d\n",&p[4][2] - &a[4][2],&p[4][2] - &a[4][2]);

	return 0;
}

解析:

在这里插入图片描述
所以&p[4][2] - &a[4][2]得到-4

注意%p是打印地址的,要算出补码,再转换成十六进制。
10000000000000000000000000000100 -4的原码
11111111111111111111111111111011 反码
1111 1111 1111 1111 1111 1111 1111 1100 补码
F F F F F F F C //四个二进制位换算成一个十六进制位
%p打印-4的地址是0xFFFFFFFC


总结

以上就是今天要分享的五道笔试题。

  • 21
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 23
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 23
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值