C语言指针 试题篇

249 阅读9分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第21天,点击查看活动详情

一、指针和数组笔试题解析

1、实例1

💨 在此之前我们先回顾一下数组名: ▶ 一、sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小 ▶ 二、&数组名,这里的数组名表示整个数组,取出的是整个数组的地址 ▶ 三、除此之外所有的数组名都表示数组首元素的地址 注:在下面实例中,我会在每条语句的后面写上正确的输出结果,并用一、二、三(代表上面的场景)来配合解释 -> 【正确答案:一/二/三】-> 更详细的解释

1.1、一维数组

#include<stdio.h>
int main()
{
	int a[] = {1,2,3,4};
	printf("%d\n",sizeof(a));//【16:一】-> 4*4=16
	printf("%d\n",sizeof(a+0));//【4/8:三】-> sizeof(首元素地址)
	printf("%d\n",sizeof(*a));//【4:三】-> sizeof(1)
	printf("%d\n",sizeof(a+1));//【4/8:三】-> sizeof(2)	
	printf("%d\n",sizeof(a[1]));//【4:三】-> sizeof(2)
	printf("%d\n",sizeof(&a));//【4/8:二】
	printf("%d\n",sizeof(*&a));//【16:二】-> &和*可以抵消
	printf("%d\n",sizeof(&a+1));//【4/8:二】-> sizeof(跳过a数组后面的数组的地址)
	printf("%d\n",sizeof(&a[0]));//【4/8:三】-> 因为[]的优先级 > &,因此sizof(&1)
	printf("%d\n",sizeof(&a[0]+1));//【4/8:三】-> sizeof(&2)
	
	return 0;
}
  • 小结

这里主要注意的是:在这里插入图片描述 这里&a是取出整个数组的地址,+1就跳过了这个数组 在这里插入图片描述

1.2、字符数组

#includ<stdio.h>
int main()
{
	char arr[] = {'a','b','c','d','e','f'};
	printf("%d\n", sizeof(arr));//【6:一】-> sizeof(数组总大小),
	printf("%d\n", sizeof(arr+0));//【4/8:三】-> sizeof(&a)
	printf("%d\n", sizeof(*arr));//【1:三】-> sizeof(a)
	printf("%d\n", sizeof(arr[1]));//【1:三】-> sizeof(b)
	printf("%d\n", sizeof(&arr));//【4/8:二】-> sizeof(数组的地址)
	printf("%d\n", sizeof(&arr+1));//【4/8:二】-> sizeof(跳过arr数组后面的数组的地址)
	printf("%d\n", sizeof(&arr[0]+1));//【4/8:三】-> sizeof(&b)
	//---------------------------------分割线-----------------------------------
	printf("%d\n", strlen(arr));//【19:随机值】-> strlen(&a),strlen在求字符串长度时是以'\0'为标志的,且不包含'\0'
	printf("%d\n", strlen(arr+0));//【19:随机值】-> strlen(&a)
	printf("%d\n", strlen(*arr));//【err:三】-> strlen(a),strlen只能传地址
	printf("%d\n", strlen(arr[1]));//【err:同上】-> strlen(b)
	printf("%d\n", strlen(&arr));//【19:随机值】-> 这里虽然计算的是地址,站在strlen的角度strlen会把数组指针的类型转换为一级指针,然后会从a的地址开始往下找
	printf("%d\n", strlen(&arr+1));//【13:随机值}-> 这里会从arr数组最后一个元素的后面往下找
	printf("%d\n", strlen(&arr[0]+1));//【18:随机值】-> 这里是从arr数组第2个元素开始往下数
	//---------------------------------分割线-----------------------------------
	char arr2[] = "abcdef";
	printf("%d\n", sizeof(arr));//【7:一】-> sizeof(数组大小)
	printf("%d\n", sizeof(arr+0));//【4/8:三】-> sizeof(&a)
	printf("%d\n", sizeof(*arr));//【1:三】-> sizeof(a)
	printf("%d\n", sizeof(arr[1]));//【1:三】-> sizeof(b)
	printf("%d\n", sizeof(&arr));//【4/8:二】-> sizeof(&数组的地址)
	printf("%d\n", sizeof(&arr+1));//【4/8:二】-> sizeof(跳过数组后的地址)
	printf("%d\n", sizeof(&arr[0]+1));//【4/8:三】sizeof(&b)

	//---------------------------------分割线-----------------------------------
	printf("%d\n" , str1en(arr));//【6:】-> strlen在计算字符串长度时,不会包含'\0'
	printf("%d\n", strlen(arr+O));//【6:】-> strlen(&a)
	printf("%d\n", strlen(*arr));//【err:】-> strlen(a)
	printlf ("%d\n", strlen(arr[1]));//【err:】-> strlen(b)
	printf("%d\n",strlen(&arr));//【6:】-> strlen(&a)
	printf("%d\n", strlen(&arr+1));//【12:随机值】-> strlen(跳过数组后的地址)
	printf("%d\n",strlen(&arr[O]+1));//【5:】-> strlen(&b)
	
	return 0;
}
  • 小结

==一、strlen在计算字符串长度时是以'\0'为标志的,且不会计算'\0'== ==二、在使用strlen去计算字符串数组时== >{'a','b','c','d','e','f'} -> a,b,c,d,e,f
"abcdef" -> a,b,c,d,e,f,0 ==三、之前我们在模拟strlen函数的时候 -> int my_strlen(const char* str)它的参数是一个指针,应该接收地址,而这里传的是一个字符== 在这里插入图片描述 所以err: 在这里插入图片描述 ==四、当strlen的参数是一个数组的地址时== 在这里插入图片描述 这里计算的是整个数组地址,在传参的过程中把数组的地址传给一级指针时,这时数组的类型就会变成char*,因为这里是strlen说了算,站在strlen的角度往后数也是一个随机值, 五、当我们比较这三条语句时: 在这里插入图片描述

1.3、指针指向的字符串

#include<stdio.h>
int main()
{
	char* p = "abcdef";
	printf("%d\n", sizeof(p));//【4/8:】-> sizeof(&a)	
	printf("%d\n", sizeof(p+1));//【4/8:】-> sizeof(&b)
	printf("%d\n", sizeof(*p));//【1:】-> sizeof(a)
	printf("%d\n", sizeof(p[0]));//【1:】->sizeof(a)
	printf("%d\n", sizeof(&p));//【4/8:】-> sizeof(p指针的地址)
	printf("%d\n", sizeof(&p+1));//【4/8:】-> sizeof(跳过p指针的地址)
	printf("%d\n", sizeof(&p[0]+1));//【4/8:】-> sizeof(&b)
	//---------------------------------分割线-----------------------------------
	printf("%d\n",strlen(p));//【6:】-> strlen(&a)	
	printf("%d\n",strlen(p+1));//【5:】-> strlen(&b)
	printf("%d\n",strlen(*p));//【err:】-> strlen(a)
	printf("%d\n", strlen(p[0]));//【err:】-> strlen(a)	
	printf("%d\n",strlen(&p));//【3:随机值】-> strlen(p指针的地址)
	printf("%d\n", strlen(&p+1));//【11:随机值】-> strlen(跳过p指针的地址)
	printf("%d\n", strlen(&p[0]+1));//【5:】-> strlen(&b)
	
	return 0;
}
  • 小结:

==一、这里关于指针和数组名有一些关联:== int arr[10] = {1,2,3,4,5,6,7,8,9,10}; int* p = arr; 可推出:arr[2] < == > *(arr + 2) < == > *(p + 2) < == > *(2 + p) < == > *(2 + arr) < == > 2[arr] ==二、当我们取指针的地址时:== 在这里插入图片描述

1.4、二维数组

#include<stdio.h>
int main()
{
	int a[3][4] = {0};
	printf("%d\n" , sizeof(a));//【48:一】-> 3*4*4=48
	printf("%d\n", sizeof(a[0][O]));//【4:三】-> sizeof(第一行数组名[0])
	printf("%d\n" , sizeof(a[0]));//【16:一】-> sizeof(第一行数组名)
	printf("%d\n" , sizeof(a[0]+1));//【4/8:三】-> 因为数组名没有单独放到sizeof里,所以sizeof(第1行第2个元素的地址)
	printf("%d\n" ,sizeof(*(a[0]+1)));//【4:三】-> sizeof(*(第1行第2个元素的地址))
	printf("%d\n" , sizeof(a+1));//【4/8:三】-> 这里a表示首元素地址(a[0]),所以a+1就是第二行数组的地址
	printf("%d\n" ,sizeof(*(a+1)));//【16:三】-> 这里计算的是第2行数组的大小
	printf("%d\n" , sizeof(&a[0]+1));//【4/8:二】-> &a[0]取出第1行数组的地址,因此它计算的是第2行数组的地址
	printf("%d\n" ,sizeof(*(&a[0]+1)));//【16:二】-> 这里计算的是第2行数组的大小
	printf ("%d\n" ,sizeof(*a));//【16:三】-> 这里表数组首元素地址a[0],因此计算的是第1行数组的大小
	printf("%d\n" ,sizeof(a[3]));//【16:一】-> 这里并没有真正的去访问a[3],而是根据a[3]的类型int[4]去计算大小  
	
	return 0;
}
  • 小结

==一、二维数组的数组名==在这里插入图片描述 ==二、sizeof去计算越界的数组== 在这里插入图片描述 这里为什么不是err? 因为sizeof内部的表达式是不计算的,所以我们不论越界多大都不会err 我们在写任何一个表达式(3+5)时它都有两个属性: 1、值属性 - 8 2、类型属性 - int 所以对于sizeof我们根本不需要知道表达式运算的值是多少,只要能够算出来结果是int类型就行了sizeof(3+5) -> 4。所以a[3]的类型是int[4]   补充(看一道例子): int main() {   short s = 5;   int a = 4;   printf("%d\n", sizeof(s = a + 6);// 2 - 这里不管是多大s说了算   printf("%s\n", s);// 5 - 证明了sizeof不会去计算()里的表达式   return 0; }

2、实例2

#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));//2,5
	return 0;
}
  • 分析:

---------------------------* (ptr - 1)---------------------------在这里插入图片描述 ---------------------------* (a + 1)--------------------------- 在这里插入图片描述

3、实例3

#include<stdio.h>
//由于还没有学习结构体,这里告知结构体的大小是20个字节
struct Test
{
	int Num;
	char* pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}*p;
//假设p的值为0x100000,如下表达式的值分别是多少?
int main()
{
	//0x1就是十六进制的1,它同十进制的1
	printf("%p\n", p + 0x1);//0x100014
	printf("%p\n", (unsigned long)p + 0x1);//0x100001
	printf("%p\n", (unsigned int*)p + 0x1);//0x100004
	return 0;
}
  • 分析

==经分析,这里考察指针类型决定了指针的运算== ---------------------------p + 0x1--------------------------- ==p的类型是struct Test,且占20个字节。而p+1跳过这个结构体(20个字节)== 0x100000 + 1 == 0x100014(需要注意的是十进制20要先转换为十六进制) ---------------------------(unsigned long)p + 0x1--------------------------- 这里主要注意的是: ==非指针类型的变量+1,那它+1加的就是1== (unsigned long)p + 0x1); 0x100000 + 1 == 0x100001   ==只有是指针类型的变量+1才跳过的是一个类型== (unsigned long*)p + 0x1); 0x100000 + 1 ==0x100004 ---------------------------(unsigned int*)p + 0x1)--------------------------- ==这里是将p强转为int无符号int类型的指针,所以它+1会跳过一个整型== 0x100000 + 1 ==0x100004

4、实例4

#include<stdio.h>
int main()
{
	int a[4] ={1,2,3,4};
	int* ptr1 = (int*)(&a + 1);//4
	int* ptr2 = (int*)((int)a + 1);//2000000
	printf("%x,%x", ptr1[-1], *ptr2);
	return 0;
}
  • 分析

---------------------------ptr1[-1]----------------------- 在这里插入图片描述 ---------------------------*ptr2--------------------------- 在这里插入图片描述

5、实例5

#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;
}
  • 分析

在这里插入图片描述

6、实例6

#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]);//FFFFFFFC,- 4
	return 0;
}
  • 分析

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

7、实例7

#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)); //10,5
	return 0;
}
  • 分析

---------------------------(ptr1 - 1)----------------------- 1、&aa取二维数组的地址 2、+1跳过二维数组,此时它的类型是int( * )[2][5] 3、再将它强制类型转换为int后赋值于int* ptr1 4、(ptr1-1)就找到了10 在这里插入图片描述 ---------------------------(ptr2 - 1)----------------------- 1、aa是首元素的地址,二维数组的首元素是第一行 2、aa+1指向第二行的数组 3、*(aa+1)就对第二行数组解引用,就相当于第二行数组的数组名    *(aa+1) == aa[1]   ==可以说 *(二维数组的首地址)就是数组名== 4、此时 *(aa+1)它的类型就是 int *,可以直接赋值于 int * ptr2 在这里插入图片描述

在这里插入图片描述

8、实例8

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

在这里插入图片描述

9、实例9

#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);//POINT
	printf("%s\n", *-- * ++cpp + 3);//ER
	printf("%s\n", *cpp[-2] + 3);//ST
	printf("%s\n", cpp[-1][-1] + 1);//EW 
	return 0;
}
  • 分析

==先了解一下:== #include<stdio.h> int main() {   int arr[] = { 1,2,3,4,5 };   int* p;   p = arr;   printf("%d\n", *++p);//2   printf("%d\n", ++p);//3 -> 对于指针类型(地址)的操作,操作的是它的地址,所以本身它会被改变   int a = 3;   printf("%d\n", a + 3);//6   printf("%d\n", a + 3);//6 -> 而对于非指针类型(地址)操作,并不是操作它的地址,所以本身它没有改变   return 0; } -----------------------------图示--------------------------------- 在这里插入图片描述 -----------------------------**++cpp--------------------------------- 在这里插入图片描述 ------------------------------- * ++cpp + 3--------------------------------- 在这里插入图片描述 -----------------------------*cpp[-2] + 3--------------------------------- 在这里插入图片描述 -----------------------------cpp[-1][-1] + 1--------------------------------- 在这里插入图片描述