当前位置: 首页 > news >正文

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

本节内容以刷题为主,大致目录:

1.一维数组

2.字符数组

3.二维数组

学完后,你将对数组有了更全面的认识


在刷关于数组的题目前,我们先认识一下数组名:

数组名的意义:表示数组首元素的地址

但是有两个例外:

(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;
}

结果展示:

代码分析:

#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;
}

 内存分局图:

总结:只要可以确定是地址,那么大小一定就是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;
}

结果展示:

代码分析:

#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;
}

内存布局图:

(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;
}

结果展示:

代码分析:

#include<stdio.h>
int main()
{char arr[] = "abcdef";//arr中的内容:a,b,c,d,e,f,\0printf("%zd\n", sizeof(arr));//整个数组大小:7字节printf("%zd\n", sizeof(arr + 0));//地址,4/8字节printf("%zd\n", sizeof(*arr));//首元素,1字节大小printf("%zd\n", sizeof(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;
}

内存分布图:

(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));

结果展示:

代码分析:

#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;
}

内存分布图:

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;
}

内存分布图: 

(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,\0printf("%d\n", strlen(arr));//从起始位置开始计算长度,为6printf("%d\n", strlen(arr + 0));//从起始位置开始计算长度,为6printf("%d\n", strlen(*arr));//*arr==a,传入的值就为97,代码报错printf("%d\n", strlen(arr[1]));//arr[1]==b,传入的值为98,代码错误printf("%d\n", strlen(&arr));//从起始位置开始计算,长度为6printf("%d\n", strlen(&arr + 1));//&arr为起始地址,&arr+1跳过了整个数组,为随机值printf("%d\n", strlen(&arr[0] + 1));//从第二个元素开始,为5return 0;
}

内存分布图:

(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));//从起始位置开始计算长度,为6printf("%d\n", strlen(p + 1));//从b位置开始计算长度,为5printf("%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位置开始计算,长度为5return 0;
}

内存分布图:

二、二维数组

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

题目:判断下列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;
}

运行结果:

代码分析:

#include<stdio.h>
int main()
{int a[3][4] = { 0 };//三行四列,一行的大小是12,整个数组是48printf("%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=16printf("%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;
}

内存布局图:

总结:有二维数组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]就表示首元素的地址

相关文章:

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

本节内容以刷题为主&#xff0c;大致目录&#xff1a; 1.一维数组 2.字符数组 3.二维数组 学完后&#xff0c;你将对数组有了更全面的认识 在刷关于数组的题目前&#xff0c;我们先认识一下数组名&#xff1a; 数组名的意义&#xff1a;表示数组首元素的地址 但是有两个例外…...

vue-router学习(四) --- 动态添加路由

我们一般使用动态添加路由都是后台会返回一个路由表前端通过调接口拿到后处理(后端处理路由)。比如不同权限显示不同的路由。 主要使用的方法就是router.addRoute 添加路由 动态路由主要通过两个函数实现。router.addRoute() 和 router.removeRoute()。它们只注册一个新的路…...

科东软件受邀参加2023国家工业软件大会,共话工业软件未来

10月28日&#xff0c;由中国自动化学会主办的2023国家工业软件大会在浙江湖州开幕。大会以“工业软件智造未来”为主题&#xff0c;一批两院院士、千余名专家学者齐聚一堂&#xff0c;共同探讨工业软件领域前沿理论和技术创新应用问题&#xff0c;共同谋划我国工业软件未来发展…...

ros启动节点的launch文件你真的会写吗?

<launch><!-- 启动节点 --><node name="lidar_data_feature_detection_node" pkg="lidar_data_feature_detection" type="lidar_data_feature_detection" output="screen" />...

AMEYA360:循序积累立体布局,北京君正实景展示AI-ISP

北京君正集成电路股份有限公司(下称“北京君正”)是国内较早深耕智能安防及泛视觉解决方案的芯片供应商之一&#xff0c;也是国内同时掌握CPU、VPU、ISP、AIE等核心技术的创新企业之一&#xff0c;自成立以来始终深耕行业&#xff0c;并持续迭代创新产品及创新方案。 在2023 CP…...

10.31 知识总结(选择器、css属性相关)

一、选择器 1.1 属性选择器 通过标签的属性来查找标签&#xff0c;标签都有属性 <div class"c1" id"d1"></div> id值和class值是每个标签都自带的属性&#xff0c;还有另外一种&#xff1a;自定义属性 <div class"c1" id"d1…...

【网络协议】聊聊TCP如何做到可靠传输的

网络是不可靠的&#xff0c;所以在TCP协议中通过各种算法等机制保证数据传输的可靠性。生活中如何保证消息可靠传输的&#xff0c;那么就是采用一发一收的方式&#xff0c;但是这样其实效率并不高&#xff0c;所以通常采用的是累计确认或者累计应答。 如何实现一个靠谱的协议&…...

记一次flask框架环境综合渗透测试

PART.01 登入过程 访问靶场地址http://101.43.22.226/?name2023&#xff0c;框架为Flask。 2. 测试存在ssti注入。 3. 直接执行以下命令。 http://101.43.22.226/?name{% for c in [].class.base.subclasses() %} {% if c.name ‘catch_warnings’ %} {% for b in c.i…...

博弈论学习笔记(2)——完全信息静态博弈

前言 这部分我们学习的是完全信息静态博弈&#xff0c;主要内容包括博弈论的基本概念、战略式博弈、Nash均衡、Nash均衡解的特性、以及Nash均衡的应用。 零、绪论 1、什么是博弈论 1&#xff09;博弈的定义 博弈论&#xff1a;研究决策主体的行为发生直接相互作用时候的决策…...

【COMP304 LEC4 LEC5】

LEC 4 1. Truth-Functionality Propositional logic 的connectives&#xff08;连接词&#xff09;are truth-functional 但是&#xff0c;有时候的描述不是true-functional的&#xff0c;比如&#xff1a;"Knowing that", "It is necessary that",&quo…...

表白墙(服务器)

目录 0.需求 1.创建Maven项目 2.给pom.xml内引入三个依赖 3.完善目录&#xff0c;并补充web.xml中的内容 4.编写代码 后端代码 ​编辑前端代码 5.引入数据库 创建message表 创建工具类 往MessageServlet类中添加方法 0.需求 前面写好了表白墙页面&#xff0c;但存…...

在 Mac 中卸载 Node.js

在 Mac 中卸载 Node.js&#xff0c;可以选择以下两种方法&#xff1a; 使用命令行卸载 Node.js 第一步&#xff1a;打开终端&#xff0c;输入以下命令显示 Node.js 的安装路径&#xff1a; which node 执行该命令后&#xff0c;会显示安装路径&#xff1a;/usr/local/bin/n…...

Hafnium构建选项及FVP模型调用

安全之安全(security)博客目录导读 目录 一、Hafnium构建选项 二、FVP模型调用 一、Hafnium构建选项 本节解释了在支持基于FF-A的SPM (SPMD位于EL3, SPMC位于S-EL1、S-EL2或EL3)的情况下进行构建时涉及的TF-A构建选项:...

第44天:前端及html、Http协议

前端 前端是所有跟用户直接打交道的都可以称之为是前端&#xff0c;比如&#xff1a;PC页面、手机页面、平板页面、汽车显示屏、大屏幕展示出来的都是前端内容。 前端的用处&#xff1a; 学了前端以后我们就可以做全栈工程师(会后端、会前端、会DB、会运维等),能够写一些简单的…...

shell_63.Linux产生信号

Linux 系统信号 信号 值 描述 1 SIGHUP 挂起&#xff08;hang up&#xff09;进程 2 SIGINT 中断&#xff08;interrupt&#xff09;进程 3 SIGQUIT 停止&#xff08;stop&#xff09;进程 9 …...

互联网摸鱼日报(2023-11-01)

互联网摸鱼日报(2023-11-01) 36氪新闻 毫末智行张凯&#xff1a;2023年高阶智能辅助驾驶市场迎来大爆发 ​撕开三星、金士顿市场&#xff0c;国产老牌存储器企业出海三年&#xff0c;营收翻三倍&#xff5c;insight全球 给医生一双“透视眼”&#xff0c;「锦瑟医疗」专注开…...

AR的光学原理?

AR智能眼镜的光学成像系统 AR眼镜的光学成像系统由微型显示屏和光学镜片组成&#xff0c;可以将其理解为智能手机的屏幕。 增强现实&#xff0c;从本质上说&#xff0c;是将设备生成的影像与现实世界进行叠加融合。这种技术基本就是通过光学镜片组件对微型显示屏幕发出的光线…...

语义分割 实例分割的异同点

语义分割和实例分割是计算机视觉领域中两个相关但不同的任务&#xff0c;它们都涉及对图像像素进行分类和标记&#xff0c;但关注的对象和目标有所不同。 目标对象&#xff1a; 语义分割&#xff1a;语义分割的目标是将图像中的每个像素标记为对应的语义类别&#xff0c;即将…...

C++学习初探---‘C++面向对象‘-继承函数重载与运算符重载

文章目录 前言继承继承是什么&#xff1f;三种访问权限的继承&#xff1a; 函数重载与运算符重载函数重载运算符重载可重载运算符&不可重载运算符 前言 第三次学习记录&#xff0c;依旧是C面向对象的内容。 继承 继承是什么&#xff1f; C中的继承是一种面向对象编程&am…...

Linux下搭建SRS服务器环境

搭建环境 Ubuntu的Linux环境srs 安装源码&#xff1a;源码地址为&#xff1a;GitHub - ossrs/srs at 3.0release 搭建步骤 下载srs源码 git clone GitHub - ossrs/srs: SRS is a simple, high-efficiency, real-time video server supporting RTMP, WebRTC, HLS, HTTP-FLV,…...

进程地址空间(比特课总结)

一、进程地址空间 1. 环境变量 1 &#xff09;⽤户级环境变量与系统级环境变量 全局属性&#xff1a;环境变量具有全局属性&#xff0c;会被⼦进程继承。例如当bash启动⼦进程时&#xff0c;环 境变量会⾃动传递给⼦进程。 本地变量限制&#xff1a;本地变量只在当前进程(ba…...

什么是EULA和DPA

文章目录 EULA&#xff08;End User License Agreement&#xff09;DPA&#xff08;Data Protection Agreement&#xff09;一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA&#xff08;End User License Agreement&#xff09; 定义&#xff1a; EULA即…...

多模态大语言模型arxiv论文略读(108)

CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题&#xff1a;CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者&#xff1a;Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...

Mobile ALOHA全身模仿学习

一、题目 Mobile ALOHA&#xff1a;通过低成本全身远程操作学习双手移动操作 传统模仿学习&#xff08;Imitation Learning&#xff09;缺点&#xff1a;聚焦与桌面操作&#xff0c;缺乏通用任务所需的移动性和灵活性 本论文优点&#xff1a;&#xff08;1&#xff09;在ALOHA…...

基于Java+MySQL实现(GUI)客户管理系统

客户资料管理系统的设计与实现 第一章 需求分析 1.1 需求总体介绍 本项目为了方便维护客户信息为了方便维护客户信息&#xff0c;对客户进行统一管理&#xff0c;可以把所有客户信息录入系统&#xff0c;进行维护和统计功能。可通过文件的方式保存相关录入数据&#xff0c;对…...

深入理解Optional:处理空指针异常

1. 使用Optional处理可能为空的集合 在Java开发中&#xff0c;集合判空是一个常见但容易出错的场景。传统方式虽然可行&#xff0c;但存在一些潜在问题&#xff1a; // 传统判空方式 if (!CollectionUtils.isEmpty(userInfoList)) {for (UserInfo userInfo : userInfoList) {…...

深度学习之模型压缩三驾马车:模型剪枝、模型量化、知识蒸馏

一、引言 在深度学习中&#xff0c;我们训练出的神经网络往往非常庞大&#xff08;比如像 ResNet、YOLOv8、Vision Transformer&#xff09;&#xff0c;虽然精度很高&#xff0c;但“太重”了&#xff0c;运行起来很慢&#xff0c;占用内存大&#xff0c;不适合部署到手机、摄…...

uniapp 集成腾讯云 IM 富媒体消息(地理位置/文件)

UniApp 集成腾讯云 IM 富媒体消息全攻略&#xff08;地理位置/文件&#xff09; 一、功能实现原理 腾讯云 IM 通过 消息扩展机制 支持富媒体类型&#xff0c;核心实现方式&#xff1a; 标准消息类型&#xff1a;直接使用 SDK 内置类型&#xff08;文件、图片等&#xff09;自…...

ZYNQ学习记录FPGA(一)ZYNQ简介

一、知识准备 1.一些术语,缩写和概念&#xff1a; 1&#xff09;ZYNQ全称&#xff1a;ZYNQ7000 All Pgrammable SoC 2&#xff09;SoC:system on chips(片上系统)&#xff0c;对比集成电路的SoB&#xff08;system on board&#xff09; 3&#xff09;ARM&#xff1a;处理器…...

如何通过git命令查看项目连接的仓库地址?

要通过 Git 命令查看项目连接的仓库地址&#xff0c;您可以使用以下几种方法&#xff1a; 1. 查看所有远程仓库地址 使用 git remote -v 命令&#xff0c;它会显示项目中配置的所有远程仓库及其对应的 URL&#xff1a; git remote -v输出示例&#xff1a; origin https://…...