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

news2024/11/27 8:37:37

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

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,\0

	printf("%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,\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;
}

内存分布图:

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

内存分布图:

二、二维数组

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

题目:判断下列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,整个数组是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;
}

内存布局图:

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1070385.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Kafka 简介之(学习之路)

正文 一、简介 1.1 概述 Kafka是最初由Linkedin公司开发&#xff0c;是一个分布式、分区的、多副本的、多订阅者&#xff0c;基于zookeeper协调的分布式日志系统&#xff08;也可以当做MQ系统&#xff09;&#xff0c;常见可以用于web/nginx日志、访问日志&#xff0c;消息服务…

挑选适合您的优秀项目管理软件

哪个项目管理软件好用&#xff0c;这全得看用户需求。有的企业项目组比较多&#xff0c;项目比较大&#xff0c;就需要重一些的软件。有的企业就是简单管理一下项目进展&#xff0c;看看工时&#xff0c;那轻量级项目管理软件就挺好用&#xff0c;因为上手很快。还有的初创小团…

《软件方法》第1章2023版连载(04)不了解ABCD工作流的危害

DDD领域驱动设计批评文集 做强化自测题获得“软件方法建模师”称号 《软件方法》各章合集 1.2 建模工作流 1.2.4 不了解ABCD的危害 1.2.4.1 思维颠倒 如果软件开发人员对以上的“A-业务建模”、“B-需求”、“C-分析”、“D-设计”工作流没有概念&#xff0c;就会把软件开…

vulnhub_Empire_LupinOne靶机渗透测试

Empire_LupinOne靶机 靶机地址&#xff1a;Empire: LupinOne ~ VulnHub 文章目录 Empire_LupinOne靶机信息收集web渗透获取权限横向移动权限提升靶机总结 信息收集 通过nmap扫描得到靶机开放22和80端口&#xff0c;进行全扫描得到了一些服务版本信息等&#xff0c;web端有ro…

GitHub基础

1、仓库是什么意思&#xff1f;仓库拥有者是谁&#xff1f; 在软件开发或版本控制系统中&#xff0c;"仓库"&#xff08;Repository&#xff09;是指存储项目代码、配置文件、文档等相关文件的地方。它可以看作是一个中央存储库&#xff0c;用于管理和跟踪项目的各个…

v-bind绑定

一、标签属性动态绑定 方式一&#xff1a; v-bind:属性名"data声明的变量名" 方式二&#xff1a;&#xff08;简写&#xff09; 将v-bind省略&#xff0c;直接 :属性名"data声明的变量" v-bind.属性名[.修饰符]"变量名、计算属性。。 对应还提供了修饰…

【Docker】简易版harbor部署

文章目录 依赖于docker-compose下载添加执行权限测试 安装harbor下载解压修改配置文件部署配置开机自启动登录验证 使用harbor登录打标签上传下载 常见问题 依赖于docker-compose 下载 curl -L “https://github.com/docker/compose/releases/download/2.22.0/docker-compose-…

第八章 排序 十三、置换-选择排序

目录 一、概括 二、例子 ​三、考点 一、概括 置换-选择排序是一种排序算法&#xff0c;它通过在未排序的元素中选择最小的元素并将其放置在已排序的部分的末尾来逐步将列表排序。具体过程如下&#xff1a; 从列表中选择最小的元素&#xff0c;并将其与列表中第一个元素交…

一文看懂光模块的工作原理

你们好&#xff0c;我的网工朋友 光模块有很多类别&#xff0c;是我们经常要用到的PHY层器件。虽然封装&#xff0c;速率&#xff0c;传输距离有所不同&#xff0c;但是其内部组成基本是一致的。 以太网交换机常用的光模块有SFP&#xff0c;GBIC&#xff0c;XFP&#xff0c;X…

【周末闲谈】“PHP是最好的语言”这个梗是怎么来的?

个人主页&#xff1a;【&#x1f60a;个人主页】 系列专栏&#xff1a;【❤️周末闲谈】 系列目录 ✨第一周 二进制VS三进制 ✨第二周 文心一言&#xff0c;模仿还是超越&#xff1f; ✨第二周 畅想AR 文章目录 系列目录前言最早的出处关于PHP语言优点缺点网络评价 总结 前言 …

kafka日志文件详解及生产常见问题总结

一、kafka的log日志梳理 日志文件是kafka根目录下的config/server.properties文件&#xff0c;配置log.dirs/usr/local/kafka/kafka-logs&#xff0c;kafka一部分数据包含当前Broker节点的消息数据(在Kafka中称为Log日志)&#xff0c;称为无状态数据&#xff0c;另外一部分存在…

Vue中如何进行网页截图与截屏

在Vue中实现网页截图与截屏功能 网页截图与截屏功能在许多Web应用程序中都非常有用。Vue.js作为一个流行的JavaScript框架&#xff0c;提供了许多工具和库来简化网页截图和截屏的实现。本文将介绍如何使用Vue来实现一个网页截图和截屏功能的示例&#xff0c;包括使用html2canv…

锁向环到底是什么?是怎么进行倍频的?

你们有没有这样一个疑问&#xff0c;就是CPU的主频怎么做到几个GHz呢&#xff1f; 每一秒要给处理器几亿个脉冲&#xff0c;就拿11代I7处理器来说&#xff0c;它的基本频率就可达2.5GHz&#xff0c;但在我们常规的认知中&#xff0c;频率的大小取决于晶振的频率&#xff0c;比…

黑马JVM总结(二十八)

&#xff08;1&#xff09;语法糖-foreach &#xff08;2&#xff09;语法糖-switch-string &#xff08;3&#xff09;语法糖-switch-enum &#xff08;4&#xff09;语法糖-枚举类 枚举类 &#xff08;5&#xff09;语法糖-twr1

云表|都有生产管理模块,MES和ERP有什么不同,该如何选择

MES和ERP是生产制造领域的两大知名系统&#xff0c;虽然早已声名鹊起&#xff0c;但仍有不少人难以明确区分两者的差异。下面将详细阐述这两个系统的不同之处。首先&#xff0c;要了解MES和ERP的定义。 MES系统&#xff1a;全称制造执行系统&#xff08;Manufacturing Executio…

【软考】8.2 编译程序基本原理/文法/正规式/有限自动机

《编译程序基本原理》 编译过程 词法分析&#xff1a; 针对单词&#xff1b;输入是字符&#xff1b;读的是字符流&#xff1b;语法分析&#xff1a; 针对语句&#xff1b;读的是记号流&#xff0c;即词法分析产生的一个个单词语义分析&#xff08;针对语句含义&#xff09; a.…

Javascript中的模块化详解

1.什么是模块化、模块化开发&#xff1f; 事实上模块化开发最终的目的是将程序划分成一个个小的结构&#xff1b; 这个结构中编写属于自己的逻辑代码&#xff0c;有自己的作用域&#xff0c;不会影响到其他的结构&#xff1b; 这个结构可以将自己希望暴露的变量、函数、对象等…

漏洞复现--时空智友企业流程化管控系统敏感信息泄露(POC)

免责声明&#xff1a; 文章中涉及的漏洞均已修复&#xff0c;敏感信息均已做打码处理&#xff0c;文章仅做经验分享用途&#xff0c;切勿当真&#xff0c;未授权的攻击属于非法行为&#xff01;文章中敏感信息均已做多层打马处理。传播、利用本文章所提供的信息而造成的任何直…

进程状态的理解

我们知道进程会有属于自己的PCB&#xff0c;便于操作系统的管理&#xff0c;而PCB结构体里面还有进程状态参数&#xff0c;类似于用一个变量标识对应的进程状态&#xff0c;就相当于将每个进程状态编号&#xff0c;而PCB中有一个变量存储当前进程状态所对应的编号&#xff0c;也…

国庆要闻回顾 | OpenAI 拟研发 AI 手机;9月以太坊上NFT销售量创2021年2月以来最低记录...

国庆期间区块链行业要闻回顾&#xff1a;产业方面&#xff0c;全国区块链行业产教融合共同体在雄安新区成立&#xff0c;巴西推出基于区块链的数字身份证&#xff0c;瑞银集团在以太坊上推出代币化货币市场基金试点&#xff0c;NASA拟在月球设立区块链数据中心以保存国家机密资…