数组笔试题详解

发布时间:2024年01月15日

数组笔试题解析

我们可以通过做题来加深我们对数组及相关知识的理解,下面的笔试题解答正确的关键在于下面这点,一定要牢记:
数组名是首元素地址,两种情况除外:
1.sizeof(数组名) , 这是这是计算整个数组的大小,单位是字节;
2.&数组名 , 得出的是整个数组的地址;

下面我们来做几组数组相关的笔试题:
例1:

#include<stdio.h>
int main()
{
	//一维数组
	int a[] = { 1,2,3,4 };
	printf("%d\n", sizeof(a));
	//a是数组名单独放在sizeof内部,计算的是整个数组的大小,单位字节,结果是16
	printf("%d\n", sizeof(a + 0));
	//a+0是数组名首元素地址,是地址就是4/8个字节(32位平台下4字节,64位平台下8字节)
	printf("%d\n", sizeof(*a));
	//*a是对首元素的解引用,拿到的是1,则大小为4字节
	printf("%d\n", sizeof(a + 1));
	//a+1是首元素地址+1,也就是数组第二个元素的地址,是地址就是4/8字节
	printf("%d\n", sizeof(a[1]));
	//a[1]是第二个元素,其大小为4字节
	printf("%d\n", sizeof(&a));
	//&a是整个数组的地址,是地址就是4/8个字节,&a的类型:int(*)[4]
	printf("%d\n", sizeof(*&a));
	//*&a是对整个数组的解引用,拿到的是整个数组的元素,结果是16字节,*&a->a->aizeof(a)
	printf("%d\n", sizeof(&a + 1));
	//&a+1跳过的是整个数组,指向的是整个数组后面的地址,是地址就是4/8字节
	printf("%d\n", sizeof(&a[0]));
	//&a[0]是对首元素取地址,是地址就是4/8字节
	printf("%d\n", sizeof(&a[0] + 1));
	//&a[0]+1是数组第二个元素的地址,是地址就是4/8字节
}

在32位平台下:
在这里插入图片描述

例2:

#include<stdio.h>
int main()
{
	//字符数组
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%d\n", sizeof(arr));
	//arr是数组名单独放在sizeof内部,计算的是整个数组的大小,结果是6字节
	printf("%d\n", sizeof(arr + 0));
	//arr是数组名没有单独放进sizeof内部,表示的是数组首元素地址,首元素地址+0,还是首元素地址
	//arr+0取的是数组的第一个元素的地址,是地址就是4/8字节
	printf("%d\n", sizeof(*arr));
	//*arr是对数组首元素的解引用,拿到的是'a',结果就是1字节
	printf("%d\n", sizeof(arr[1]));
	//arr[1]是第二个元素的大小,结果就是1字节
	printf("%d\n", sizeof(&arr));
	//&arr是整个元素的地址,是地址就是4/8字节
	printf("%d\n", sizeof(&arr + 1));
	//&arr+1是跳过整个数组,指向的是整个数组后面的地址,结果就是4/8字节
	printf("%d\n\n", sizeof(&arr[0] + 1));
	//&arr[0]+1是数组的第二个元素的地址,是地址就是4/8字节


	printf("%d\n", strlen(arr));
	//strlen只有遇到\0才结束,所以是随机值
	printf("%d\n", strlen(arr + 0));
	//arr+0是首元素地址,结果是4/8字节
	printf("%d\n", strlen(*arr));
	//strlen(*arr)->strlen('a')->strlne(97),非法访问内存,error
	printf("%d\n", strlen(arr[1]));
	//srtlen(arr[1])->strlen('b')->strlen(98),非法访问内存,error
	printf("%d\n", strlen(&arr));
	//随机值
	printf("%d\n", strlen(&arr + 1));
	//随机值
	printf("%d\n", strlen(&arr[0] + 1));
	//不知道什么时候会遇到\0,所以是随机值

}

例3:

#include<stdio.h>
int main()
{
	char arr[] = "abcdef";
	//"abcdef" -> "abcdef\0"
	printf("%d\n", sizeof(arr));
	//arr是数组名,单独放在sizeof内部,计算的是整个数组的大小,结果是7字节
	printf("%d\n", sizeof(arr + 0));
	//arr+0是数组首元素的地址,结果是4/8
	printf("%d\n", sizeof(*arr));
	//*arr是对数组首元素的解引用,结果是1字节
	printf("%d\n", sizeof(arr[1]));
	//arr[1]是数组的第二个元素,结果是1字节
	printf("%d\n", sizeof(&arr));
	//&arr是整个数组的地址,结果是4/8字节
	printf("%d\n", sizeof(&arr + 1));
	//&arr+1是跳过整个数组,指向的是整个数组后的地址,结果是4/8字节
	printf("%d\n", sizeof(&arr[0] + 1));
	//&arr[0]+1是第二个元素的地址,结果就是4/8字节


	printf("%d\n", strlen(arr));
	//strlen(arr)是整个数组的个数,因为strlen不计算\0的大小,所以结果是6个
	printf("%d\n", strlen(arr + 0));
	//arr+0是第一个元素,向后数共6个
	printf("%d\n", strlen(*arr));
	//strlen(*arr)->strlen('a')->strlen(97),非法访问内存,error
	printf("%d\n", strlen(arr[1]));
	//strlen(*arr)->strlen('b')->strlen(98),非法访问内存,error
	printf("%d\n", strlen(&arr));
	//&arr是整个数组个数,结果是6个
	printf("%d\n", strlen(&arr + 1));
	//随机值,因为是跳过整个数组,指向的是整个数组后面,不知道什么时候会遇到\0,所以个数是不确定的
	printf("%d\n", strlen(&arr[0] + 1));
	//&arr[0]+1是第二个元素,向后数共5个,结果是5个
}

通过上面的笔试题我们知道:
1.strlen是求字符串的长度,计算的是’\0’之前的字符的个数,不包含’\0’
2.sizeof 的计算是包含’\0’的大小

例4:

#include<stdio.h>
int main()
{
	char* p = "abcdef";
	printf("%d\n", sizeof(p));
	//p是指针变量,存放的是地址,是地址就是4/8字节
	printf("%d\n", sizeof(p + 1));
	//p+1是第二个元素的地址,是地址就是4/8字节
	printf("%d\n", sizeof(*p));
	//*p是'a',计算的是'a'的大小,结果是1字节
	printf("%d\n", sizeof(p[0]));
	//p[0]是'a',计算的是'a'的大小,结果是1字节
	printf("%d\n", sizeof(&p));
	//&p是二级指针,是指针也是地址,指针大小就是4/8字节
	printf("%d\n", sizeof(&p + 1));
	//&p+1是跳过整个数组的地址,指向整个数组后的地址就,结果就是4/8字节
	printf("%d\n", sizeof(&p[0] + 1));
	//&p[0]+1是数组第二个元素的地址,是地址就是4/8字节


	printf("%d\n", strlen(p));
	//strlen(p)就是求p所指向的元素直到\0之前的元素个数,结果就是6个
	printf("%d\n", strlen(p + 1));
	//p+1是第二个元素,从第二个元素开始到\0之前的字符个数,结果是5个
	printf("%d\n", strlen(*p));
	//*p是a,strlen('a')->strlen(97),非法访问内存,error
	printf("%d\n", strlen(p[0]));
	//p[0]是a,strlen('a')->strlen(97),非法访问内存,error
	printf("%d\n", strlen(&p));
	//&p是p这个指针变量的起始地址,从起始地址到\0,不确定中间元素个数,所以是随机值
	printf("%d\n", strlen(&p + 1));
	//&p+1,跳过整个指针变量,指向的是整个指针变量后的地址,不确定什么时候能遇到\0,所以是随机值
	printf("%d\n", strlen(&p[0] + 1));
	//&p[0]+1是第二个元素,结果是5个
}

字符串里面默认有个’\0’,字符里面没有默认的’\0’,字符串是双引号括起来的,字符是单引号括起来的;
sizeof计算时会包含’\0’的计算,而strlen不会包含’\0’的计算;

例5:

#include<stdio.h>
int main()
{
	//二维数组
	int a[3][4] = { 0 };
	printf("%d\n", sizeof(a));
	//a是数组名,单独放在sizeof内部,是计算整个数组的大小,结果是48字节
	printf("%d\n", sizeof(a[0][0]));
	//a[0][0]是第一行第一个元素,结果是4字节
	printf("%d\n", sizeof(a[0]));
	//a[0]是单独放在sizeof内部的,a[0]代表第一行的数组名,计算的是第一行的大小,结果是16字节
	printf("%d\n", sizeof(a[0] + 1));
	//a[0]+1没有单独放在sizeof内部,所以a[0]就是a[0][0],+1则就是第一行第二个元素的地址,结果就是4/8字节
	printf("%d\n", sizeof(*(a[0] + 1)));
	//*(a[0]+1)就是对第一行第二个元素解引用,结果是4字节
	printf("%d\n", sizeof(a + 1));
	//a+1是第二行的地址,结果就是4/8字节
	printf("%d\n", sizeof(*(a + 1)));
	//*(a+1)就是对第二行解引用,结果就是16字节
	printf("%d\n", sizeof(&a[0] + 1));
	//&a[0]是取第一行的地址,+1就是第二行的地址,结果就是4/8字节
	printf("%d\n", sizeof(*(&a[0] + 1)));
	//*(&a[0]+1)是对第二行的解引用,结果就是16字节
	printf("%d\n", sizeof(*a));
	//*a是对第一行的 解引用,结果是16字节
	printf("%d\n", sizeof(a[3])); // 不会真实访问,会根据类型确定字节大小
	//a[3]是第四行的数组名,计算第四行的大小,结果是16字节
}

总结:

1.数组名是首元素地址,两种情况除外:
①sizeof(数组名) , 这是这是计算整个数组的大小,单位是字节;
②&数组名 , 得出的是整个数组的地址;

2.strlen是求字符串的长度,计算的是’\0’之前的字符的个数,不包含’\0’;

3.sizeof 的计算是包含’\0’的大小;

4.字符串里面默认有个’\0’,字符里面没有默认的’\0’,字符串是双引号括起来的,字符是单引号括起来的;

5.sizeof不会真实访问空间,是通过变量的类型来计算占用空间大小的;

6.放在sizeof内部的表达式不会参与真实计算;

7.sizeof是在编译期间进行的;真实参与计算的表达式实在运行期间执行的;

文章来源:https://blog.csdn.net/weixin_65245159/article/details/128667213
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。