【进阶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]就表示首元素的地址

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

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

相关文章

强化学习环境 - robogym - 学习 - 3

强化学习环境 - robogym - 学习 - 3 文章目录 强化学习环境 - robogym - 学习 - 3项目地址为什么选择 robogymObservation - 观测信息Action - 动作信息Initialization - 初始状态设置 项目地址 https://github.com/openai/robogym 为什么选择 robogym 自己的项目需要做一些机…

软考 系统架构设计师系列知识点之软件架构风格(7)

接前一篇文章&#xff1a;软考 系统架构设计师系列知识点之软件架构风格&#xff08;6&#xff09; 这个十一注定是一个不能放松、保持“紧”的十一。由于报名了全国计算机技术与软件专业技术资格&#xff08;水平&#xff09;考试&#xff0c;11月4号就要考试&#xff0c;因此…

SQL创建与删除索引

索引创建、删除与使用&#xff1a; 1.1 create方式创建索引&#xff1a;CREATE [UNIQUE – 唯一索引 | FULLTEXT – 全文索引 ] INDEX index_name ON table_name – 不指定唯一或全文时默认普通索引 (column1[(length) [DESC|ASC]] [,column2,…]) – 可以对多列建立组合索引 …

【JS原型链,常见的修改原型对象的方法有哪些?】

原型链 什么是原型链修改原型对象的方法1. 修改原型对象上的属性和方法2. 重写原型对象3. 使用Object.create创建新的原型对象4. 使用Object.setPrototypeOf修改原型链 什么是原型链 JavaScript中每个对象都有一个内部属性[[Prototype]]指向它的原型对象&#xff0c;原型对象也…

acwing算法基础之基础算法--高精度乘法算法

目录 1 知识点2 模板 1 知识点 大数乘以小数 大数的每一位与小数相乘 2 模板 //A是大数&#xff0c;b是小数 //最后去除高位上的0 vector<int> mul(vector<int>& A, int b) {vector<int> C;int t 0;for (int i 0; i < A.size() || t; i) {if (i …

8.2 JUC - 7.线程安全集合类概述

目录 一、遗留的线程安全集合二、使用 Collections 装饰的线程安全集合三、java.util.concurrent.* 包下的对象 线程安全集合类可以分为三大类&#xff1a; 一、遗留的线程安全集合 Hashtable &#xff0c; Vector 二、使用 Collections 装饰的线程安全集合 Collections.sy…

salesforce点击classic按钮调用lightning组件

创建按钮&#xff0c;选择visualforce 页面&#xff0c;然后创建visualforce页面。 how to Call a Lightning component from a Classic custom button You can definitely call a Lightning Component from a Classic custom button. However, the process is not straightfo…

微信小程序使用路由传参和传对象的方法

近期在做微信小程序开发&#xff0c;在页面跳转时&#xff0c;需要携带参数到下一个页面&#xff0c;尤其是将对象传入页面。为了方便重温&#xff0c;特此记录。 路由传字符串参数 原始页面 传递字符串参数比较简单。路由跳转有两种方式&#xff0c;一种是通过navigator组件…

企业AI工程化之路:如何实现高效、低成本、高质量的落地?

MLOps工程实践 概述面临挑战目的内容简介读者对象专家推荐目录 写在末尾&#xff1a; 主页传送门&#xff1a;&#x1f4c0; 传送 概述 作为计算机科学的一个重要领域&#xff0c;机器学习也是目前人工智能领域非常活跃的分支之一。机器学习通过分析海量数据、总结规律&#x…

Java异常:基本概念、分类和处理

Java异常&#xff1a;基本概念、分类和处理 在Java编程中&#xff0c;异常处理是一个非常重要的部分。了解如何识别、处理和避免异常对于编写健壮、可维护的代码至关重要。本文将介绍Java异常的基本概念、分类和处理方法&#xff0c;并通过简单的代码示例进行说明。 一、什么…

Flask实现注册登录模块

&#x1f64c;秋名山码民的主页 &#x1f602;oi退役选手&#xff0c;Java、大数据、单片机、IoT均有所涉猎&#xff0c;热爱技术&#xff0c;技术无罪 &#x1f389;欢迎关注&#x1f50e;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; 获取源码&#xff0c;添加WX 目录 前言1.…

AIGC(生成式AI)试用 8 -- 曾经的难题

长假&#xff0c;远离电脑、远离手机、远离社交。 阴雨连绵&#xff0c;望着窗外发呆&#xff0c;AIGC为何物&#xff1f;有什么问题要问AIGC&#xff1f;AIGC可以代替我来发呆&#xff0c;还是可是为我空出时间发呆&#xff1f; 如果可以替代我发呆&#xff0c;要我何…

机器学习之SGD, Batch, and Mini Batch的简单介绍

文章目录 总述SGD(Stochastic Gradient Descent)(随机梯度下降&#xff09;Batch &#xff08;批量&#xff09;mini Batch (迷你批量&#xff09; 总述 SGD, Batch, and Mini Batch是可用于神经网络的监督学习计算权重更新的方案&#xff0c;即∆wij。 SGD(Stochastic Gradi…

el-tree中插入图标并且带提示信息

<template><div class"left"><!-- default-expanded-keys 默认展开 --><!-- expand-on-click-node 只有点击箭头才会展开树 --><el-tree :data"list" :props"defaultProps" node-click"handleNodeClick" :…

c语言进阶部分详解(详细解析字符串常用函数,并进行模拟实现(下))

上篇文章介绍了一些常用的字符串函数&#xff0c;大家可以跳转过去浏览一下&#xff1a;c语言进阶部分详解&#xff08;详细解析字符串常用函数&#xff0c;并进行模拟实现&#xff08;上&#xff09;&#xff09;_总之就是非常唔姆的博客-CSDN博客 今天接着来介绍一些&#x…

Linux防火墙之--SNAT和DNAT

1.SNAT是什么 SNAT又称源地址转换。源地址转换是内网地址向外访问时&#xff0c;发起访问的内网ip地址转换为指定的ip地址&#xff08;可指定具体的服务以及相应的端口或端口范围&#xff09;&#xff0c;这可以使内网中使用保留ip地址的主机访问外部网络&#xff0c;即内网的多…

docker-compose一键部署mysql

1.创建安装目录 mnt为硬盘挂载目录&#xff0c;根据实际情况修改 mkdir -p /mnt/mysql cd /mnt/mysql vim docker-compose.yml2.编写docker-compose.yml version: 3.1 services:db:image: mysql:5.7 #mysql版本volumes:- ./data/db:/var/lib/mysql #数据文件- ./etc/my.cnf:/…

腾讯云轻量和CVM有啥区别?怎么选择服务器配置?

腾讯云轻量服务器和云服务器有什么区别&#xff1f;为什么轻量应用服务器价格便宜&#xff1f;是因为轻量服务器CPU内存性能比云服务器CVM性能差吗&#xff1f;轻量应用服务器适合中小企业或个人开发者搭建企业官网、博客论坛、微信小程序或开发测试环境&#xff0c;云服务器CV…

RK3568 适配MIPI屏AYF070R1140H07-M

首先需要修改mipi屏的初始化参数,一般初始化参数由厂家提供 找到对应的DTS文件(因为不同的板卡厂商对应的DTS不一样,但是设置是相同的),将参数填写到这里面,这里整理好了基本参数,只要是使用同个型号的屏幕就可以正常使用 &dsi1_panel {status = "okay"…

TDengine OSS 与 qStudio 实现无缝协同,革新数据分析和管理方式

在数字化转型如火如荼的当下&#xff0c;海量爆发的时序数据处理成为转型成功的关键因素之一。为了帮助社区用户更好地进行数据分析和管理&#xff0c;丰富可视化解决方案的多样性&#xff0c;我们将开源的时序数据库&#xff08;Time Series Database&#xff09; TDengine OS…