前言
前段时间收藏了一些指针笔试题,简单和困难模式都有。打算暑假来做一做,却拖了很久,在收藏夹吃灰了,最近我夜观天象,今日又晴空万里,伸出手指一算,发现,今天是个做题的好天气,我连忙翻出了被遗忘在收藏夹里的练习题,今日就来攻破它!
一、指针笔试题
笔试题一
代码如下:
#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
总结
以上就是今天要分享的五道笔试题。