笔试题解析【指针和数组上篇】

前言

本章适合学完指针和数组的家人们巩固和提升,下面有一道笔试题,如果你能快速并且准确地说出他的答案,那么我相信你对于C语言指针和数组的理论知识已经基本没什么问题了,本章你可以跳过。留下来的家人们,让我们一起攻克他。

//二维数组
int a[3][4] = {0};
printf("%d\n",sizeof(a));
printf("%d\n",sizeof(a[0][0]));
printf("%d\n",sizeof(a[0]));
printf("%d\n",sizeof(a[0]+1));
printf("%d\n",sizeof(*(a[0]+1)));
printf("%d\n",sizeof(a+1));
printf("%d\n",sizeof(*(a+1)));
printf("%d\n",sizeof(&a[0]+1));
printf("%d\n",sizeof(*(&a[0]+1)));
printf("%d\n",sizeof(*a));
printf("%d\n",sizeof(a[3]));

1.一维数组

知识点:

关于数组名
数组名是数组首元素的地址
但是有2个例外:
1. sizeof(数组名) - 数组名表示整个数组,计算的是整个数组的大小,单位是字节
2. &数组名 - 数组名也表示整个数组,取出的是整个数组的地址
除了这个2个例外,你见到的所有的数组名都表示首元素的地址

题目:

//一维数组
int a[] = {1,2,3,4};
printf("%d\n",sizeof(a));
printf("%d\n",sizeof(a+0));
printf("%d\n",sizeof(*a));
printf("%d\n",sizeof(a+1));
printf("%d\n",sizeof(a[1]));
printf("%d\n",sizeof(&a));
printf("%d\n",sizeof(*&a));
printf("%d\n",sizeof(&a+1));
printf("%d\n",sizeof(&a[0]));
printf("%d\n",sizeof(&a[0]+1));

1.1 sizeof(a);

int a[] = {1,2,3,4};//定义一个一维数组,他里面有四个元素

在这里插入图片描述

printf("%d\n",sizeof(a));// 16

此时a作为数组名单独存放在sizeof内部,计算的是数组的总大小,单位是字节。
在这里插入图片描述

1.2 sizeof(a+0);

printf("%d\n",sizeof(a+0));// 4/8

a并非单独存放在sizeof内部,也没有&,所以数组名a就是数组首元素的地址,a+0还是数组首元素的地址,是地址大小就是4/8个字节(32位平台下是4个字节,64位平台是8个字节)。
在这里插入图片描述

1.3 sizeof(*a);

printf("%d\n",sizeof(*a));// 4/8

a是首元素的地址,*a就是首元素,sizeof(*a)就算的就是首元素的大小。
a是int* 类型的,*a是int类型的。
在这里插入图片描述

1.4 sizeof(a+1);

printf("%d\n",sizeof(a+1));// 4/8

a是首元素的地址,a+1是第二个元素的地址,sizeof(a+1)计算的是指针的大小 - 4/8
a是int* 类型的,a+1跳过一个int类型。
在这里插入图片描述

1.5 sizeof(a[1]);

printf("%d\n",sizeof(a[1])); // 4

a[1]就是数组的第二个元素,sizeof(a[1])的大小 - 4个字节。
a[1] == *(a+1)
在这里插入图片描述

1.6 sizeof(&a);

printf("%d\n",sizeof(&a));// 4/8

&a取出的数组的地址,数组的地址,也是地址,sizeof(&a)就是 4/8 个字节。
在这里插入图片描述

1.7 sizeof(*&a);

printf("%d\n",sizeof(*&a));// 16

&a是数组的地址,是数组指针类型,*&a是都数组指针解引用,访问一个数组的大小。
sizeof(*&a) ⇒ sizeof(a) == 16
sizeof(&*a) == sizeof(a+0)
在这里插入图片描述

1.8 sizeof(&a+1);

printf("%d\n",sizeof(&a+1));// 4/8 

&a数组的地址,&a+1跳过整个数组,&a+1还是地址,是 4/8 个字节。
在这里插入图片描述

1.9 sizeof(&a[0]);

printf("%d\n",sizeof(&a[0]));// 4/8

a[0]是数组的第一个元素,&a[0]是第一个元素的地址,是 4/8 个字节。
在这里插入图片描述

1.10 sizeof(&a[0]+1);

printf("%d\n",sizeof(&a[0]+1));// 4/8

数组名a先和 [0] 结合,&a[0]就是首元素的地址,&a[0]+1就是数组第二个元素的地址,是地址就是4/8个字节。
在这里插入图片描述

代码运行结果

在这里插入图片描述

2.字符数组

知识点:

sizeof 是计算对象或者类型创建的对象所占内存空间的大小,单位是字节
sizeof 是操作符,不是函数

strlen 求字符串长度的,计算的是字符串中\0之前出现的字符的个数
       统计到\0为止,如果没有看到\0,会继续往后找
strlen 是库函数

2.1题目一

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));
printf("%d\n", sizeof(arr+0));
printf("%d\n", sizeof(*arr));
printf("%d\n", sizeof(arr[1]));
printf("%d\n", sizeof(&arr));
printf("%d\n", sizeof(&arr+1));
printf("%d\n", sizeof(&arr[0]+1));

printf("%d\n", strlen(arr));
printf("%d\n", strlen(arr+0));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr));
printf("%d\n", strlen(&arr+1));
printf("%d\n", strlen(&arr[0]+1));

分析:
在这里插入图片描述

2.1.1 sizeof(arr) && strlen(arr);

printf("%d\n", sizeof(arr));// 6
printf("%d\n", strlen(arr));// 随机值

sizeof:此时arr作为数组名单独存放在sizeof内部,计算的是数组的总大小,单位是字节。
strlen:arr是数组名,但是没有放在sizeof内部,也没&,arr就是首元素的地址,strlen得到arr后,从arr数组首元素的地方开始计算字符串的长度,直到直到\0,但是arr数组中没有\0,arr内存的后边是否有\0,在什么位置是不确定的,所以\0之前出现了多少个字符是随机的。

在这里插入图片描述

2.1.2 sizeof(arr+0) && strlen(arr+0);

printf("%d\n", sizeof(arr+0));// 4/8
printf("%d\n", strlen(arr+0));// 随机值

sizeof:arr作为数组名没有单独存放在sizeof内部,也没有&arr,所以此时arr作为数组名是首元素的地址,arr+0还是首元素的地址,是地址就是4/8个字节,无论他是char类型还是int或者其他类型。
strlen:arr是数组首元素的地址,arr+0还是首元素的地址,所以是随机值。
在这里插入图片描述

2.1.3 sizeof(*arr) && strlen(*arr);

printf("%d\n", sizeof(*arr));// 1
printf("%d\n", strlen(*arr));// err 会非法访问内存

sizeof:arr是首元素的地址,*arr就是首元素,sizeof计算的是首元素的大小,是1字节。
strlen:arr是数组首元素的地址,*arr 是首元素 - ‘a’ - 97,strlen就把‘a’的ASCII码值 97 当成了地址。
在这里插入图片描述

2.1.4 sizeof(arr[1]) && strlen(arr[1]);

printf("%d\n", sizeof(arr[1]));// 1
printf("%d\n", strlen(arr[1]));// err 非法访问

sizeof:arr[1]是数组的第二个元素,sizeof(arr[1])计算的是第二个元素的大小,1个字节。
strlen:arr[1]是数组的第二个元素,strlen将arr[1]的ASCII值当做地址,造成非法访问。
在这里插入图片描述

2.1.5 sizeof(&arr) && strlen(&arr);

printf("%d\n", sizeof(&arr));// 4/8
printf("%d\n", strlen(&arr));// 随机值

sizeof:&arr是数组的地址,是地址,就是4/8个字节。
strlen:&arr是数组的地址,数组的地址也是指向数组起始位置,和第一个案例一样。
在这里插入图片描述

2.1.6 sizeof(&arr+1) && strlen(&arr+1);

printf("%d\n", sizeof(&arr+1));// 4/8
printf("%d\n", strlen(&arr+1));// 随机值

sizeof:&arr是数组的地址,&arr+1跳过的是数组的地址,是地址就是4/8个字节。
strlen:随机值。
在这里插入图片描述

2.1.7 sizeof(&arr[0]+1) && strlen(&arr[0]+1);

printf("%d\n", sizeof(&arr[0]+1));// 4/8
printf("%d\n", strlen(&arr[0]+1));// 随机值

sizeof:数组名arr先和[0]结合,&arr[0]就是首元素的地址,&arr[0]+1就是数组第二个元素的地址,是地址就是4/8个字节。
strlen: 随机值。
在这里插入图片描述

代码运行结果:

在这里插入图片描述

2.2 题目二

char arr[] = "abcdef";
printf("%d\n", sizeof(arr));
printf("%d\n", sizeof(arr+0));
printf("%d\n", sizeof(*arr));
printf("%d\n", sizeof(arr[1]));
printf("%d\n", sizeof(&arr));
printf("%d\n", sizeof(&arr+1));
printf("%d\n", sizeof(&arr[0]+1));

printf("%d\n", strlen(arr));
printf("%d\n", strlen(arr+0));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr));
printf("%d\n", strlen(&arr+1));
printf("%d\n", strlen(&arr[0]+1));

分析:
在这里插入图片描述

2.2.1 sizeof(arr) && strlen(arr);

printf("%d\n", sizeof(arr));// 7
printf("%d\n", strlen(arr));// 6

sizeof:此时arr作为数组名单独存放在sizeof内部,计算的是数组的总大小,单位是字节。
strlen:arr是数组名,但是没有放在sizeof内部,也没&,arr就是首元素的地址,strlen得到arr后,从arr数组首元素的地方开始计算字符串的长度,直到直到\0。
在这里插入图片描述

2.2.2 sizeof(arr+0) && strlen(arr+0);

printf("%d\n", sizeof(arr+0));// 4/8
printf("%d\n", strlen(arr+0));// 6

sizeof:arr作为数组名没有单独存放在sizeof内部,也没有&arr,所以此时arr作为数组名是首元素的地址,arr+0还是首元素的地址,是地址就是4/8个字节,无论他是char类型还是int或者其他类型。
strlen:arr是数组首元素的地址,arr+0还是首元素的地址。
在这里插入图片描述

2.2.3 sizeof(*arr) && strlen(*arr);

printf("%d\n", sizeof(*arr));//1
printf("%d\n", strlen(*arr));// err

sizeof:数组名arr既没有单独放在sizeof内部,也没有&arr,所以此时arr作为数组名的首元素地址,*arr就是首元素,大小是一个字节。
strlen:arr是数组首元素的地址,*arr 是首元素 - ‘a’ - 97,strlen就把‘a’的ASCII码值 97 当成了地址。
在这里插入图片描述

2.2.4 sizeof(arr[1]) && strlen(arr[1]);

printf("%d\n", sizeof(arr[1]));// 1
printf("%d\n", sizeof(arr[1]));// err

sizeof:arr[1]是数组的第二个元素,sizeof(arr[1])计算的是第二个元素的大小,1个字节。
strlen:arr[1]是数组的第二个元素,strlen将arr[1]的ASCII值当做地址,造成非法访问。
在这里插入图片描述

2.2.5 sizeof(&arr) && strlen(&arr);

printf("%d\n", sizeof(&arr));// 4/8
printf("%d\n", sizeof(&arr));// 6

sizeof:&arr是数组的地址,是地址,就是4/8个字节。
strlen:&arr是数组的地址,数组的地址也是指向数组起始位置,向后找’\0’。
在这里插入图片描述

2.2.6 sizeof(&arr+1) && strlen(&arr+1);

printf("%d\n", sizeof(&arr+1));// 4/8
printf("%d\n", sizeof(&arr+1));//随机值

sizeof:&arr是数组的地址,&arr+1跳过整个数组的地址,是地址就是4/8个字节。
strlen:&arr是数组的地址,&arr+1跳过整个数组的地址,在数组后面找’\0’,随机值。
在这里插入图片描述

2.2.7 sizeof(&arr[0]+1) && strlen(&arr[0]+1);

printf("%d\n", sizeof(&arr[0]+1));// 4/8
printf("%d\n", sizeof(&arr[0]+1));// 5

sizeof:数组名arr先和[0]结合,&arr[0]就是首元素的地址,&arr[0]+1就是数组第二个元素的地址,是地址就是4/8个字节。
strlen:数组名arr先和[0]结合,&arr[0]就是首元素的地址,&arr[0]+1就是数组第二个元素的地址,从数组的第二个元素往后找’\0’。
在这里插入图片描述

代码运行结果:

在这里插入图片描述

3.二维数组

//二维数组
int a[3][4] = {0};
printf("%d\n",sizeof(a));
printf("%d\n",sizeof(a[0][0]));
printf("%d\n",sizeof(a[0]));
printf("%d\n",sizeof(a[0]+1));
printf("%d\n",sizeof(*(a[0]+1)));
printf("%d\n",sizeof(a+1));
printf("%d\n",sizeof(*(a+1)));
printf("%d\n",sizeof(&a[0]+1));
printf("%d\n",sizeof(*(&a[0]+1)));
printf("%d\n",sizeof(*a));
printf("%d\n",sizeof(a[3]));

分析:
在这里插入图片描述

3.1 sizeof(a);

printf("%d\n",sizeof(a));// 48

a是二维数组的数组名,数组名单独放在sizeof内部,计算的是数组的总大小,单位是字节。
在这里插入图片描述

3.2 sizeof(a[0][0]);

printf("%d\n",sizeof(a[0][0]));// 4

a[0][0]是一个整型元素,大小是4个字节。
在这里插入图片描述

3.3 sizeof(a[0]);

printf("%d\n",sizeof(a[0]));// 16

把二维数组的每一行看做一维数组的时候,a[0]是第一行的数组名,第一行的数组名单独放在sizeof内部,计算的是第一行的总大小,单位是字节。
在这里插入图片描述

3.4 sizeof(a[0]+1);

printf("%d\n",sizeof(a[0]+1));// 4/8

a[0]虽然是第一行的数组名,但是并非单独放在sizeof内部,a[0]作为第一行的数组名并非表示整个第一行这个数组,a[0]就是第一行首元素的地址,a[0]–> &a[0][0],a[0]+1,跳过一个int,是a[0][1]的地址。
在这里插入图片描述

3.5 sizeof(*(a[0]+1));

printf("%d\n",sizeof(*(a[0]+1)));// 4

a[0]+1是第一行第二个元素的地址,所以*(a[0]+1)就是a[0][1],大小是4个字节。
在这里插入图片描述

3.6 sizeof(a+1);

printf("%d\n",sizeof(a+1));// 4/8

a是二维数组的数组名,没单独放在sizeof内部,也没有&,所以a就是数组首元素的地址,二维数组,我们把它想象成一维数组,它的第一个元素就是二维数组的第一行,a就是第一行的地址,a+1 是第二行的地址,是地址,大小就是 4/8 个字节。
在这里插入图片描述

3.7 sizeof(*(a+1));

printf("%d\n",sizeof(*(a+1)));// 16

a+1是第二行的地址,(a+1) 找到的就是第二行,sizeof((a + 1))计算的就是第二行的大小。
在这里插入图片描述

3.8 sizeof(&a[0]+1);

printf("%d\n",sizeof(&a[0]+1));// 4/8

&a[0]是第一行的地址,&a[0]+1就是第二行的地址,sizeof(&a[0] + 1)计算的第二行地址大小,是地址就是4/8个字节。
在这里插入图片描述

3.9 sizeof(*(&a[0]+1));

printf("%d\n",sizeof(*(&a[0]+1)));// 16

&a[0] + 1是第二行的地址,*(&a[0] + 1)拿到的就是第二行,大小就是16个字节。
在这里插入图片描述

3.10 sizeof(*a);

printf("%d\n",sizeof(*a));// 16

a表示首元素的地址,就是第一行的地址 - &a[0],*a - 拿到的就是第一行 - 大小就是16个字节,*a -> *(a+0) -> a[0]。
在这里插入图片描述

3.11 sizeof(a[3]);

printf("%d\n",sizeof(a[3]));// 16

a[3]是二维数组的第4行,虽然没有第四行,但是类型能够确定,大小就是确定的。大小就是一行的大小,单位是字节 - 16
能够分析出 a[3]的类型是:int [4]
在这里插入图片描述

代码运行结果:

在这里插入图片描述

总结

数组名的意义:

1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
3. 除此之外所有的数组名都表示首元素的地址。

结语:
感谢大家看到这里,如有错误,希望各位大佬多多指正!
祝大家心情愉悦!
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值