【进阶C语言】数组笔试题解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 数组名的意义:表示数组首元素的地址但是有两个例外:(1)sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。单位:字节。(2)&数组名,这里的数组名表示整个数组,取出的是整个数组的大小。

一、一维数组

1.判断下列sizeof计算的大小

(1)一维数组

#include<stdio.h>
int main()
{
  int a[] = { 1,2,3,4 };//一维数组
  printf("%zd\n", sizeof(a));
  printf("%zd\n", sizeof(a + 0));
  printf("%zd\n", sizeof(*a));
  printf("%zd\n", sizeof(a + 1));
  printf("%zd\n", sizeof(a[1]));
  printf("%zd\n", sizeof(&a));
  printf("%zd\n", sizeof(*&a));
  printf("%zd\n", sizeof(&a + 1));
  printf("%zd\n", sizeof(&a[0]));
  printf("%zd\n", sizeof(&a[0] + 1));
  return 0;
}

结果展示:

image.png

代码分析:

#include<stdio.h>
int main()
{
  int a[] = { 1,2,3,4 };//一维数组
  //整形数组,每个元素4个字节,整个数组为16字节
  printf("%zd\n", sizeof(a));
  //数组名直接放在sizeof内部,表示整个数组的大小
  printf("%zd\n", sizeof(a + 0));
  //数组名没有单独放在sizeof内部,表示数组首元素的地址。大小就是4/8字节
  printf("%zd\n", sizeof(*a));
  //a就是首元素地址。*a==*(a+0)==a[0];所以*a就算首元素,大小为4字节
  printf("%zd\n", sizeof(a + 1));
  //a为首元素地址,+1跳过一个元素的地址,就算第二个元素的地址。a+1==&a[1]
  //只要是地址,就算4/8字节
  printf("%zd\n", sizeof(a[1]));
  //a[1]表示第二个元素,大小就是4字节
  printf("%zd\n", sizeof(&a));
  //&a,取出的是地址,只要是地址,就是4/8字节
  printf("%zd\n", sizeof(*&a));
  //*与&操作相互抵消。sizeof(*&a)==sizeof(a),为整个数组元素的大小==16字节
  printf("%zd\n", sizeof(&a + 1));
  //&a的结果是地址,+1操作后还是地址,那就是4/8字节
  printf("%zd\n", sizeof(&a[0]));
  //a[0]是第一个元素,&a[0]表示取出第一个元素的地址,地址就算4/8字节
  printf("%zd\n", sizeof(&a[0] + 1));
  //&a[0]是首元素的地址,&a[0]+1就是第二个元素的地址,大小4/8个字节
  return 0;
}

内存分局图:

image.png

总结:只要可以确定是地址,那么大小一定就是4/8字节

(2)字符数组

#include<stdio.h>
int main()
{
  char arr[] = { 'a','b','c','d','e','f' };
  printf("%zd\n", sizeof(arr));
  printf("%zd\n", sizeof(arr + 0));
  printf("%zd\n", sizeof(*arr));
  printf("%zd\n", sizeof(arr[1]));
  printf("%zd\n", sizeof(&arr));
  printf("%zd\n", sizeof(&arr + 1));
  printf("%zd\n", sizeof(&arr[0] + 1));
  return 0;
}

结果展示:

image.png

代码分析:

#include<stdio.h>
int main()
{
  char arr[] = { 'a','b','c','d','e','f' };
  //字符数组,每个元素大小1字节,整个数组大小6字节
  printf("%zd\n", sizeof(arr));
  //计算的是整个数组的大小,为6字节
  printf("%zd\n", sizeof(arr + 0));
  //计算的是地址,大小为4/8字节
  printf("%zd\n", sizeof(*arr));
  //*arr为第一个元素,大小为1字节
  printf("%zd\n", sizeof(arr[1]));
  //arr[1]表示第二个元素,大小为1字节
  printf("%zd\n", sizeof(&arr));
  //地址,4/8字节
  printf("%zd\n", sizeof(&arr + 1));
  //地址,4/8字节
  printf("%zd\n", sizeof(&arr[0] + 1));
  //地址,4/8字节
  return 0;
}

内存布局图:

image.png

(3)字符串数组

#include<stdio.h>
int main()
{
  char arr[] = "abcdef";
  printf("%zd\n", sizeof(arr));
  printf("%zd\n", sizeof(arr + 0));
  printf("%zd\n", sizeof(*arr));
  printf("%zd\n", sizeof(arr[1]));
  printf("%zd\n", sizeof(&arr));
  printf("%zd\n", sizeof(&arr + 1));
  printf("%zd\n", sizeof(&arr[0] + 1));
  return 0;
}

结果展示:

image.png

代码分析:

内存分布图:

image.png

(4)单单字符串

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

结果展示:

image.png

代码分析:

#include<stdio.h>
int main()
{
  char* p = "abcdef";
  //p指向a的地址
  printf("%zd\n", sizeof(p));
  //p是指针,存放首字符a的地址,4/8字节
  printf("%zd\n", sizeof(p + 1));
  //p+1为字符b的地址,4/8字节
  printf("%zd\n", sizeof(*p));
  //*p==p[0],拿到的是第一个字符:a,为1字节
  printf("%zd\n", sizeof(p[0]));
  //求的是字符的大小,1字节
  printf("%zd\n", sizeof(&p));
  //地址,4/8字节
  printf("%zd\n", sizeof(&p + 1));
  //地址,4/8字节
  printf("%zd\n", sizeof(&p[0] + 1));
  //地址,4/8字节
  return 0;
}

内存分布图:

image.png

2.判断下列strlen计算的大小

知识点:strlen是计算字符串的长度(个数)的函数。统计的是在字符串中\0之前出现的字符的个数

(1)不带\0的字符数组

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[] = { 'a','b','c','d','e','f' };
  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));
  return 0;
}

因为\0是strlen结束的标志,没有\0则不能正常计算

所以准确的来说,上述代码都是错误的。

错误原因解析:

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[] = { 'a','b','c','d','e','f' };
  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));
  //随机值。地址
  return 0;
}

内存分布图:

image.png

(2)自带\0的字符串数组

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[] = "abcdef";
  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));
  return 0;
}

代码分析:

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[] = "abcdef";
  //arr中的数据:a,b,c,d,e,f,\0
  printf("%d\n", strlen(arr));
  //从起始位置开始计算长度,为6
  printf("%d\n", strlen(arr + 0));
  //从起始位置开始计算长度,为6
  printf("%d\n", strlen(*arr));
  //*arr==a,传入的值就为97,代码报错
  printf("%d\n", strlen(arr[1]));
  //arr[1]==b,传入的值为98,代码错误
  printf("%d\n", strlen(&arr));
  //从起始位置开始计算,长度为6
  printf("%d\n", strlen(&arr + 1));
  //&arr为起始地址,&arr+1跳过了整个数组,为随机值
  printf("%d\n", strlen(&arr[0] + 1));
  //从第二个元素开始,为5
  return 0;
}

内存分布图:

image.png

(3)带\0的纯字符串

#include<stdio.h>
#include<string.h>
int main()
{
  char* p = "abcdef";
  printf("%d\n", strlen(p));
  printf("%d\n", strlen(p + 1));
  printf("%d\n", strlen(*p));
  printf("%d\n", strlen(p[0]));
  printf("%d\n", strlen(&p));
  printf("%d\n", strlen(&p + 1));
  printf("%d\n", strlen(&p[0] + 1));
  return 0;
}

代码分析:

#include<stdio.h>
#include<string.h>
int main()
{
  char* p = "abcdef";
  printf("%d\n", strlen(p));
  //从起始位置开始计算长度,为6
  printf("%d\n", strlen(p + 1));
  //从b位置开始计算长度,为5
  printf("%d\n", strlen(*p));
  //*p==a,传入97,代码报错
  printf("%d\n", strlen(p[0]));
  //代码报错
  printf("%d\n", strlen(&p));
  //&p为指针变量的地址,为随机值
  printf("%d\n", strlen(&p + 1));
  //&p+1跳过了该字符串,为随机值
  printf("%d\n", strlen(&p[0] + 1));
  //从b位置开始计算,长度为5
  return 0;
}

内存分布图:

image.png

二、二维数组

前言:二维数组的数组名同样是首元素地址,不过二维数组的首元素是第一行元素;而且依旧有两个例外。

题目:判断下列sizeof计算的大小

#include<stdio.h>
int main()
{
  int a[3][4] = { 0 };
  printf("%zd\n", sizeof(a));
  printf("%zd\n", sizeof(a[0][0]));
  printf("%zd\n", sizeof(a[0]));
  printf("%zd\n", sizeof(a[0] + 1));
  printf("%zd\n", sizeof(*(a[0] + 1)));
  printf("%zd\n", sizeof(a + 1));
  printf("%zd\n", sizeof(*(a + 1)));
  printf("%zd\n", sizeof(&a[0] + 1));
  printf("%zd\n", sizeof(*(&a[0] + 1)));
  printf("%zd\n", sizeof(*a));
  printf("%zd\n", sizeof(a[3]));
  return 0;
}

运行结果:

image.png

代码分析:

#include<stdio.h>
int main()
{
  int a[3][4] = { 0 };
  //三行四列,一行的大小是12,整个数组是48
  printf("%zd\n", sizeof(a));
  //数组名单独放在里面,计算的是整个数组的大小
  printf("%zd\n", sizeof(a[0][0]));
  //a[0][0]表示第一行第一列的元素,计算的是元素的大小,4字节
  printf("%zd\n", sizeof(a[0]));
  //a[0]表示第一行的地址,也可以称为第一行这个数组的数组名
  //sizeof(arr[0])计算的就是第一行整个一维数组的全部大小,16字节
  printf("%zd\n", sizeof(a[0] + 1));
  //a[0]没有单独存放,a[0]==&a[0][0],也表示第一行第一个元素的地址
  //所以a[0]+1==&a[0][1],地址就是:4/8字节
  printf("%zd\n", sizeof(*(a[0] + 1)));
  //a[0]+1为第二行元素的数组名,也是第二行元素的首元素,4字节
  printf("%zd\n", sizeof(a + 1));
  //a没有单独放在sizeof内部,所以表示第一行的地址
  //所以a+1是第二行的地址,4/8字节
  printf("%zd\n", sizeof(*(a + 1)));
  //a+1为第二行的地址,*(a+1)就是第二行
  //*(a+1)==a[1],第二行元素的总大小=4*4=16
  printf("%zd\n", sizeof(&a[0] + 1));
  //&a[0]是第一行的地址,&a[0]+1是第二行的地址
  //地址就是4/8字节
  printf("%zd\n", sizeof(*(&a[0] + 1)));
  //*(&a[0]+1)==a[1],计算的是第二行元素的大小,为16字节
  printf("%zd\n", sizeof(*a));
  //a没有单独放在sizeof内部,所以是第一行的地址
  //*a==a[0],计算的是第一行的元素,为16字节
  printf("%zd\n", sizeof(a[3]));
  //a[3]表示第四行元素,但是没有第四行元素
  //sizeof计算的只是类型大小,a[3]和a[0]或a[1]一样,都表示某一行元素
  //虽然没有第四行,但是跟a[0]的类型一样,都是四个int元素的数组
  return 0;
}

内存布局图:

image.png

总结:有二维数组a[3][4]


(1)a、a[0]、a[1]、a[2]都表示数组名


(2)a是二维数组的数组名,a[0]、a[1]、a[2]分别是第一、二、三行的数组名。


(3)第一、第二和第三行数组又可以称为一个一维数组


(4)a、a[0]、a[1]、a[2]单独放在sizeof内部或者&数组名,就表示整个数组


(5)没有像(4)那种,a、a[0]、a[1]、a[2]就表示首元素的地址


相关文章
|
5天前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
28 4
|
1月前
|
存储 编译器 C语言
【c语言】数组
本文介绍了数组的基本概念及一维和二维数组的创建、初始化、使用方法及其在内存中的存储形式。一维数组通过下标访问元素,支持初始化和动态输入输出。二维数组则通过行和列的下标访问元素,同样支持初始化和动态输入输出。此外,还简要介绍了C99标准中的变长数组,允许在运行时根据变量创建数组,但不能初始化。
37 6
|
1月前
|
人工智能 前端开发 JavaScript
拿下奇怪的前端报错(一):报错信息是一个看不懂的数字数组Buffer(475) [Uint8Array],让AI大模型帮忙解析
本文介绍了前端开发中遇到的奇怪报错问题,特别是当错误信息不明确时的处理方法。作者分享了自己通过还原代码、试错等方式解决问题的经验,并以一个Vue3+TypeScript项目的构建失败为例,详细解析了如何从错误信息中定位问题,最终通过解读错误信息中的ASCII码找到了具体的错误文件。文章强调了基础知识的重要性,并鼓励读者遇到类似问题时不要慌张,耐心分析。
|
1月前
|
存储 人工智能 BI
C语言:数组的分类
C语言中的数组分为一维数组、多维数组和字符串数组。一维数组是最基本的形式,用于存储一系列相同类型的元素;多维数组则可以看作是一维数组的数组,常用于矩阵运算等场景;字符串数组则是以字符为元素的一维数组,专门用于处理文本数据。
|
1月前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
1月前
|
存储 C语言
C语言:一维数组的不初始化、部分初始化、完全初始化的不同点
C语言中一维数组的初始化有三种情况:不初始化时,数组元素的值是随机的;部分初始化时,未指定的元素会被自动赋值为0;完全初始化时,所有元素都被赋予了初始值。
|
1月前
|
C语言
C语言数组
C语言数组
20 0
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
34 3
|
11天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
27 6
|
1月前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
36 10

推荐镜像

更多