大家好,我是c语言boom家宝,今天为大家带来的是c语言指针内容在大厂笔试中的真题讲解,希望能让初学者对指针有更深入的理解。
ps:如有侵权,请私信联系,立刻删除。
真题一:
答案:2 , 5 。
解析:字母a是数组名,所以&a就表示整个数组,但是指向的是数组首元素的地址,而&a+1过后就是跳过一整个数组大小的字节,也就是指向这个数组末尾元素的后面的地址。然后再把它解引用并强制转换成int*类型然后用prt指针指向它的地址。为什么是强转成int*类型呢?他不还是指向地址吗?因为这个本身的类型是int(*)[5],是一个数组指针类型,对他解引用过后能操作的范围是整个数组的大小。现在强制转换成int*类型过后再进行解引用操作就只能操作一个整型的范围了。所以如果打印*(ptr-1)就是打印的5,而a表示首元素的地址,a+1就表示第二个元素的地址,*(a+1)解引用就打印的是2。所以最后答案就是2,5。如图:
真题二:
答案: 0x100014,0x100001,0x100004
解析:这道题很简单,甚至都不需要画图帮助理解。第一个问题p的值为0x100000,p + 0x1就等于加一,这个加一表示着跳过一个结构体的大小,而结构体类型的变量大小是20字节,所以就等于加了一个十进制的20,换成16进制是0x000014,所以加起来就是0x100014.第二个问题要把它强制转换成unsigned long类型,无符号长整型的加减法就是直接的加减法,加上1就表示值的大小加1,所以输出结果就是0x100001。第三个问题是强制转换成unsigned int*类型,加1就是跳过4个字节,所以输出结果就是0x100004。
真题三:假设当前机器是小端机器,请计算下面结果
答案:4, 2000000
解析:因为题目有告知当前代码是在小端机器上运行的,所以我们可以很快速的画出下面的图:
&a表示整个数组的地址,&a+1就表示跳过整个数组,即指向数组最后一个元素的后面的地址。同样的,把&a+1强制转换成int*类型再赋值给ptr1以后,那么ptr1也是指向和&a+1一样的地址,不过类型不是int(*)[],而是int*,对ptr1只能操作一个整型的变量。ptr[-1]即表示*(ptr1-1),而ptr1-1只能跳过一个整型,所以解引用过后表示的是0x00000004。但是以%x打印,编译器是不会打印出16进制的前面为0的部分,所以第一个问题打印的答案是4。第二问:先把a强制转换成int类型,那么加一就是单纯的值加一,本身a表示的是首元素的地址,指向01,加一跳过一个整型,指向00。然后强转成int*类型。又因为是小端机器,所以对ptr2进行解引用就表示的是0x02000000,%x的方式打印就是20000000。如图:
真题四:
答案: 1
解析:首先,如果我们不够仔细的话,就会掉入陷阱。题目的二维数组大括号里面是用的三个小括号分隔开的,那么他其实是逗号表达式,逗号表达式就是从左到右最后的结果就是最终的结果。所以这个数组实际上存储的内容是{1,3,5,0,0,0}。p=a[0],所以p指向的是a[0][0]的地址,也就是指向的1 。而打印的内容是p[0],也就是*(p+0)=*p的值,也就是1了。
真题五:
答案:FFFFFFFC , -4
解析:首先,a是数组指针,类型是int(*)[5],而p的类型是int(*)[4]。现在要把a的值赋给p,只能强行转换了。a指向的是数组首元素的地址,赋给p过后,p也指向了数组首元素的地址。然后我们可以简单画一个图:
因为p是int(*)[4]的类型,所以在上图中就是以4为一组的划分,a是以5为一组的划分。通过上图,我们也可以很轻易的找到&p[4][2]和&a[4][2]的地址。而地址是由低到高变化的,所以上图中左边是低地址,右边是高地址,故而&p[4][2] - &a[4][2]用%d的形式打印就是-4,因为他们如上图他们之间隔了4个,又是小的减去大的,所以是负四。而用%p形式打印,我们就需要把-4转换成原码,为1000000000000000000000000100,在转换成反码就是11111111111111111111111111111011,补码就是反码加一等于11111111111111111111111111111100,而%p打印才不管你是不是无符号位,直接把你这个补码当成地址打印了。所以我们只需要把内存中的这个补码转成16进制打印就行,4个2进制位换算成1个16进制,所以最后答案就是FFFFFFFC。
真题6:
答案:10 , 5
解析:&aa表示首元素地址,加一过后跳过整个数组,即指向数组最后一个元素的后面的地址。强转成int*类型过后再给ptr1,并且打印*(ptr1 - 1)的值,这个问题在前面已经出现过很多次了,相信大家能直接说出答案就是10 。而ptr2是由*(aa+1)强转过来的,aa+ 1表示的是数组aa[1][ ]的内容,指向的是aa[1][0]的地址,也就是6,所以ptr2 - 1就是指向的5的地址,解引用过后打印的就是5