【C语言】深入学习数组

在这里插入图片描述

👦个人主页:Weraphael
✍🏻作者简介:目前正在回炉重造C语言(2023暑假)
✈️专栏:【C语言航路】
🐋 希望大家多多支持,咱一起进步!😁
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注😍


目录

  • 一、一维数组
      • 1.1 数组的创建和初始化
      • 1.2 一维数组的使用
      • 1.3 一维数组在内存中的存储(重点)
  • 二、二维数组
      • 2.1 二维数组的创建和初始化
      • 2.2 二维数组的使用
      • 2.3 二维数组在内存中的存储
  • 三、数组越界
  • 四、数组名
      • 4.1 验证数组名是首元素地址
      • 4.2 数组名不是首元素地址的两个例外

一、一维数组

数组是一组 相同类型元素的集合

1.1 数组的创建和初始化

  • 数组的创建:
type_t  arr_name [const_n];//type_t --- 数组的元素类型
//arr_name --- 数组名
//const_n --- 是一个常量表达式(不能使用变量),用来指定数组元素个数
// 但在C99规定:const_n可以使用变量,但是不能初始化!
  • 数组的初始化:数组的初始化是指在创建数组的同时给数组的内容一些合理初始值。

【例如】

int arr1[10] = { 1,2,3,4,5,6 };
int arr2[] = { 1,2,3,4,5,6 };
int arr3[5] = { 1,2,3,4,5 };
int arr4[3] = { 'a','b','c' };
int arr5[] = { 'a','b','c' };
int arr6[] = "abcdef";

  1. 对于第一种未满10个元素的叫 不完全初始化(剩余的元素默认初始化为0
  2. []内也可以不指定元素个数,这时编译器会根据初始化的内容来确定数组的元素个数。
  3. 对于字符数组,需要用单引号,它的不完全初始化是'\0'ASCII码值是0);对于字符串数组,则用双引号,而它的不完全初始化(剩余的元素默认初始化也是'\0'
  4. 若有一个全局变量的整型数组且未被初始化,它的元素默认为0。

​看看以下程序:

在这里插入图片描述

  • 若要打印字符串数组,就会打印出'\0'往前的内容,因为字符串的结束标志是'\0'
  • 但要打印字符数组,除了打印出初始化的内容,因为找不到结束标志'\0',则后面打印出随机值

1.2 一维数组的使用

  • []:下标引用操作符。它是用来数组访问的操作符。
  • 数组元素是通过下标访问的,且下标从0开始的

【样例】

打印1~10

在这里插入图片描述

还可以倒序输出

在这里插入图片描述

或者还可以跳着打印

在这里插入图片描述

1.3 一维数组在内存中的存储(重点)

怎么知道一维数组在内存中的存储呢?可以把地址打印出来观察

在这里插入图片描述

通过观察发现:由于数组的元素都是int类型,因此地址之间差4个字节,并且都是连续的,所以可以得出以下结论:

  1. 一维数组在内存中是连续存放的。
  2. 随着数组下标的增长,地址是由低到高变化的。

接下来通过图来加以分析:

在这里插入图片描述

所以,通过以上两个结论,只要得到了数组首元素的地址,就能顺藤摸瓜找出所有数组的地址。

举个例子:

在这里插入图片描述

解释一下*(p+i),举个例子,当i=0时,p+i还是p,此时指向首元素的地址,通过解引用,找到了数组元素1

以往我们打印数组的地址都是用&arr[i]来访问,所以为了确保真实性,可以验证指针p+i的地址与&arr[i]的地址:

在这里插入图片描述

显然是一模一样的!!

二、二维数组

若想知道二维数组的元素个数,只需将行与列相乘即可。

2.1 二维数组的创建和初始化

  • 二维数组的创建
int arr[3][4]  //表示3行4列
char arr[3][5]  //表示3行5列
double arr[2][4]  //表示2行3列
  • 二维数组的初始化
  1. 完全初始化

在这里插入图片描述

  1. 不完全初始化

在这里插入图片描述

不完全初始化的二维数组后面依然补0

  1. 假如我想在一个整型数组3行4列中,第一行放1,2,第二行放3,4,第三行放5,6,这就要牵扯另一种初始化方式:可以将行当作一维数组

在这里插入图片描述

在这里插入图片描述

  1. 注意:二维数组如果有初始化,行可以省略,列不能省略!!!
int arr[][4] = {0};

2.2 二维数组的使用

二维数组的使用也是通过 下标 的方式来访问

下面是一个三行四列的整型数组

在这里插入图片描述

假设我要打印4,而4对应则是arr[1][1]

在这里插入图片描述

或者还能打印这整个数组

在这里插入图片描述

2.3 二维数组在内存中的存储

怎么知道二维数组在内存中的存储呢?可以同样可以把地址打印出来观察

在这里插入图片描述

观察发现:地址之间还是差4个字节,所以得出结论:二维数组和一维数组一样都是连续存储的

所以二维数组的排列方式可以变成如下图所示:

在这里插入图片描述

既然二维数组是连续存放的,且二维数组的排列方式也像一维数组。因此,我们可以把二维数组看成12个元素的一维数组。所以,只要得到了数组首元素的地址,就能顺藤摸瓜找出所有数组的地址。

例如:

在这里插入图片描述

  • 补充

刚刚讲过可以将行当作一维数组,而在访问一维数组时,我们都会通过一个变量来访问下标,如下图,我们可以将三行分别看作下标为0 1 2 3 4的一维数组,对于第一、二、三行来说,行都是不变的,变的只有列,所以可以这样理解,arr[0]arr[1]arr[2]分别当作第一行、第二行和第三行的数组名。 ​​​​​

在这里插入图片描述

所以也能这样打印数组:

#include<stdio.h>
int main()
{int arr[3][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 };//整个大小(48)/第一行的大小(16)= 3for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++){//第一行的大小(16)/第一行第一列的大小(4)=4for (int j = 0; j < sizeof(arr[0]) / sizeof(arr[0][0]); j++){printf("%d ", arr[i][j]);}}return 0;
}

三、数组越界

  1. 首先数组的下标是有范围限制的。
  2. 数组的下规定是从0开始的,如果数组有n个元素,最后一个元素的下标就是n-1。所以数组的下标如果小于0 ,或者大于n-1,就是数组越界访问了,超出了数组合法空间的访问。
  3. C 语言本身是不做数组下标的越界检查,编译器也不一定报错,但是编译器不报错,并不意味着程序就是正确的,所以程序员写代码时,最好自己做越界的检查

比如下面的代码:

在这里插入图片描述

注意: 二维数组的行和列也可能存在越界。

四、数组名

4.1 验证数组名是首元素地址

在这里插入图片描述

通过对比其地址,可见数组名就是首元素地址。

4.2 数组名不是首元素地址的两个例外

  1. sizeof(数组名)

在这里插入图片描述

刚刚验证了数组是首元素地址,一个地址的大小无非是8字节(x64环境)或者4字节(x86环境),而上面的代码却打印了40个字节,这是什么原因呢?这就是其中一个例外:sizeof内部单独放一个数组名时,这里的数组名表示整个数组,计算是整个数组的大小,单位是字节。

  1. &数组名

在这里插入图片描述

这里的数组名也表示整个数组,&数组名取出的是整个数组的地址。在这里,你们肯定发现了,&arr打印的不也是首元素的地址吗?这里我画图为大家分析:

在这里插入图片描述

虽然它们的地址是一样的,但是意义不一样,举个例子:

在这里插入图片描述

数组名本质上和首元素地址一样,如果是数组名+1或者是首元素地址+1,他们都是跳过对应类型的大小;但由于&数组名取出的是整个数组的地址,因此&数组名+1,跳过的是整个数组的大小。

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

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

相关文章

多模态系列论文--BEiT-3 详细解析

论文地址&#xff1a;Image as a Foreign Language: BEIT Pretraining for All Vision and Vision-Language Tasks 论文代码&#xff1a;BEiT-3 BEiT-3 1 引言&#xff1a;Big Convergence&#xff08;大一统&#xff09;2 BEIT-3预训练框架3 下游任务实现框架4 实验效果5 总结…

系统提权与数据窃取

系统提权与数据窃取 一、后渗透二、后渗透的技术2.1、提权2.2、横向移动2.3、持久化2.4、数据窃取2.5、漏洞利用 三、meterpreter后渗透3.1、靶机权限提升3.1.1、windows内核/服务漏洞3.1.2、bypassUAC3.1.2.1、bypassuac 进程注入3.1.2.2、bypassuac_injection 内存注入3.1.2.…

基于Tensorflow和Keras实现卷积神经网络CNN并进行猫狗识别

文章目录 一、环境配置1、安装Anaconda2、配置TensorFlow、Keras 二、猫狗数据集分类建模3.1 猫狗图像预处理3.2 猫狗分类的实例——基准模型3.1 构建神经网络3.2 配置优化器3.3 图片格式转化3.4 训练模型3.5 保存模型3.6 可视化 三、数据增强四、dropout 层五、参考资料 一、环…

第41节:cesium 闪烁效果-熊出没(含源码+视频)

结果示例: 点的闪烁:1.逐渐放大后消失;2.点闪烁。 图的闪烁:熊出没,含自定义显示文字效果。 完整源码: <template><div class="viewer"><vc-viewer @ready="ready" :logo="false">

国内几款常用热门音频功放芯片-低功耗、高保真

音频功放芯片&#xff0c;又称为音频功率放大器芯片&#xff0c;是指一种将音频信号转换成线性的输出功率的集成电路芯片&#xff0c;在音频功放领域中一类是传统意义上的模拟功放&#xff1b;另一类是数字功放&#xff0c;它们都可以实现模拟信号到数字信号的转换。 随着智能…

Redis 入门指南

Redis 入门指南 目录 Redis 入门指南 Redis 基本概念 Redis 常用命令 字符串String操作命令&#xff1a; 哈希Hash操作命令&#xff1a; 列表list操作命令 集合set操作命令&#xff1a; 有序集合sorted set 操作命令 通用命令 在Java中操作 Redis 1.Jedis 2.Spring …

Spring Boot 中的 @EnableDiscoveryClient 注解

Spring Boot 中的 EnableDiscoveryClient 注解 Spring Boot 是一个快速开发 Spring 应用程序的框架&#xff0c;它提供了一些基础设施&#xff0c;使得我们可以快速地开发出高效、可靠的应用程序。其中&#xff0c;EnableDiscoveryClient 注解是 Spring Boot 中一个非常重要的…

抖音矩阵源码搭建开发技术部署分析

目录 一、 什么是抖音矩阵&#xff1f;源码搭建开发注意事项&#xff1f; 1. 抖音矩阵概述 2. 源码搭建开发注意事项&#xff1a; 二、 使用步骤及开发代码展示 一、 什么是抖音矩阵&#xff1f;源码搭建开发注意事项&#xff1f; 1. 抖音矩阵概述 首先&#xff0c;抖音账…

编码和调制

编码与调制 消息是以二进制的形式存放在数据当中的&#xff0c;这种数据的表现形式是信号&#xff0c;而信源发出的原始信号就叫做基带信号&#xff0c;基带信号又可以分为数字基带信号和模拟基带信号。 信号需要在信道中进行传输&#xff0c;信道分为模拟信道和数字信…

Vue操作Cookie

这里用的js-cookie。 很简单&#xff0c;安装一下就可以使用了。 npm install js-cookie --save 使用示例&#xff1a; import Cookies from js-cookieexport default {name: YourComponent,methods: {setCookie() {// 设置一个名为 name 的cookie&#xff0c;值为 value&a…

谈谈——互联网生活中的隐私保护

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

通信算法之173: 物理层峰均比PAPT

一. SC-FDMA技术 OFDM技术&#xff0c;能有效克服宽带通信系统中的频率选择性衰落&#xff0c;降低传输符号间干扰。但4G LTE标准&#xff0c;OFDMA只用在其下行传输&#xff0c;上行端采用了单载波频分复用技术&#xff08;single carrier frequency domain multiple access S…