【C语言初阶】C语言数组基础:从定义到遍历的全面指南

📝个人主页🌹:Eternity._
⏩收录专栏⏪:C语言 “ 登神长阶 ”
🤡往期回顾🤡:C语言函数
🌹🌹期待您的关注 🌹🌹

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

❀数组

  • 📒1. 什么是数组?
    • 🌸数组的特点
    • 🌺数组的应用
  • 📜2. 一维数组
    • 🌈数组的创建
    • 🌞数组的初始化
    • 🌙数组的使用
    • ⭐数组在内存中的存储
  • 📚3. 二维数组
    • 🌈数组的创建
    • 🌞数组的初始化
    • 🌙数组的使用
    • ⭐数组在内存中的存储
  • 📙4. 数组越界
  • 📝5. 数组作为函数参数
    • ⛰️数组作为函数参数错误设计
    • 🏞️数组名
    • 🌄正确设计
  • 📖6. 总结


🔍前言: 在编程的世界里,数据结构是构建复杂应用程序的基石,而数组则是这些基石中最基础且最不可或缺的一种。C语言,作为一门历史悠久且广泛应用于系统编程、嵌入式开发等领域的编程语言,其数组的概念与操作更是每一位C语言学习者必须掌握的核心技能

数组,简而言之,是一种连续存储相同类型数据的集合。它允许我们通过索引(或下标)快速访问其中的元素,无论是读取还是修改,都极为高效。C语言中的数组不仅支持一维形式,还可以轻松扩展到多维,为处理复杂数据提供了极大的便利

本文旨在全面而深入地介绍C语言数组的基本概念、声明与初始化、访问与遍历、以及多维数组的应用等关键内容。通过理论讲解与实例演示相结合的方式,我们将逐步揭开C语言数组的神秘面纱,帮助读者建立扎实的数组知识基础,并掌握在实际编程中灵活应用数组的技巧

让我们一同踏上这段充满挑战与收获的C语言数组之旅吧!


📒1. 什么是数组?

数组(Array)是一种基础的数据结构,用于在计算机内存中连续存储相同类型的数据。它允许通过索引(或下标)来访问这些数据元素,索引通常是从0开始的。数组中的每个元素可以通过计算偏移量来快速定位,这使得数组在访问元素时非常高效


🌸数组的特点

  • 类型一致性: 数组中的所有元素都必须是相同的数据类型。这意味着,如果你有一个整型数组,那么数组中的所有元素都必须是整数
  • 固定大小: 在大多数编程语言中,数组的大小在声明时就必须确定,并且之后不能改变(尽管有些语言支持动态数组或类似的数据结构,如C++的std::vector或Python的列表,它们提供了动态大小的数组功能)。然而,在C99标准中,C语言引入了变长数组(VLA),其大小可以在运行时确定,但这仍然受到栈大小等限制
  • 索引访问: 数组中的元素可以通过索引来访问,索引通常是从0开始的。例如,在C语言中,如果你有一个名为arr的数组,并且你想要访问它的第一个元素,你可以使用arr[0]
  • 内存连续性: 数组中的元素在内存中是连续存储的。这意味着,如果你知道数组中某个元素的地址,你可以很容易地计算出数组中其他元素的地址

🌺数组的应用

  • 存储和处理一系列的数据,如学生的成绩、商品的库存量等
  • 作为函数参数传递数据集合
  • 实现算法,如排序、搜索等
  • 字符串处理,因为字符串在C语言中是通过字符数组来实现的
  • 表示多维数据结构,如矩阵和表格

尽管数组是编程中非常基础且强大的工具,但它们也有一些局限性,比如大小固定(对于传统数组而言)和类型单一。因此,在需要更灵活的数据结构时,程序员可能会选择使用其他数据结构,如链表、树或图等。然而,对于许多常见的编程任务来说,数组仍然是首选的数据结构之一


📜2. 一维数组

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


🌈数组的创建

数组的创建方式:

type_t arr_name [const_n];
// type_t 是指数组的元素类型
// const_n 是一个常量表达式,用来指定数组的大小

数组创建代码示例 (C语言):

//代码1
int arr1[10];//代码2
int count = 10;
int arr2[count];//代码3
char arr3[10];
float arr4[1];
double arr5[20];

注意:我们来看一下代码二,数组创建,在C99标准之前, [ ] 中要给一个常量才可以,不能使用变量。在C99标准支持了变长数组的概念,数组的大小可以使用变量指定,但是数组不能初始化


🌞数组的初始化

数组的初始化是指,在创建数组的同时给数组的内容一些合理初始值(初始化)


数组的初始化:

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

数组在创建的时候如果想不指定数组的确定的大小就得初始化。数组的元素个数根据初始化的内容来确定

注意:要区分以下这两种数组初始化

char arr1[] = "abc";
char arr2[3] = { 'a','b','c' };

🌙数组的使用

对于数组的使用我们之前介绍了一个操作符: [] ,下标引用操作符。它其实就数组访问的操作符


数组的使用:

int main()
{int arr[10] = { 0 };//数组的不完全初始化//计算数组的元素个数int sz = sizeof(arr) / sizeof(arr[0]);//对数组内容赋值,数组是使用下标来访问的,下标从0开始。所以:int i = 0;//做下标// for循环遍历数组for (i = 0; i < sz; i++){arr[i] = i;}//输出数组的内容i = 0;// while循环遍历数组while (i < sz){printf("%d ", arr[i]);i++;}return 0;
}

通过示例:

  • 数组是使用下标来访问的,下标是从0开始
  • 数组的大小可以通过计算得到
int arr[10];
int sz = sizeof(arr)/sizeof(arr[0]);

在这里插入图片描述


⭐数组在内存中的存储

代码示例 (C语言):

int main()
{int arr[10] = { 0 };int i = 0;int sz = sizeof(arr) / sizeof(arr[0]);for (i = 0; i < sz; ++i){printf("&arr[%d] = %p\n", i, &arr[i]);}return 0;
}

在这里插入图片描述

注意:仔细观察输出的结果,我们知道,随着数组下标的增长,元素的地址,也在有规律的递增

由此可以得出结论:数组在内存中是连续存放的
在这里插入图片描述


📚3. 二维数组

二维数组(也称为矩阵)是计算机科学中常用的数据结构,用于存储具有两个维度的数据集合。简单来说,它是一个数组的数组,即每个元素本身也是一个数组。二维数组在图像处理、游戏开发、数据分析、科学计算等领域有广泛应用


🌈数组的创建

数组的创建方式:

type_t arr_name [const_n][const_m];
// type_t 是指数组的元素类型
// const_n 表示行的大小
// const_m 表示列的大小//数组创建
int arr[3][4];
char arr[3][5];
double arr[2][4];

🌞数组的初始化

数组的初始化方式:

//数组初始化
int arr[3][4] = {1,2,3,4};
int arr[3][4] = {{1,2},{4,5}};
int arr[][4] = {{2,3},{4,5}};
// 二维数组如果有初始化,行可以省略,列不能省略

🌙数组的使用

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

数组的遍历:

int main()
{int arr[3][4] = { 0 };int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 4; j++){arr[i][j] = i * 4 + j;}}for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 4; j++){printf("%d ", arr[i][j]);}printf("\n");}return 0;
}

在这里插入图片描述


⭐数组在内存中的存储

我们来打印以下二维数组的地址看看

内存中的存储:

int main()
{int arr[3][4];int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 4; j++){printf("&arr[%d][%d] = %p\n", i, j, &arr[i][j]);}}return 0;
}

在这里插入图片描述
我们可以看出来,其实二维数组在内存中也是连续存储的

在这里插入图片描述


📙4. 数组越界

数组越界(Array Bounds Violation 或 Index Out of Bounds)是编程中常见的一个错误,主要发生在尝试访问数组时,使用的索引超出了数组的有效范围。在大多数编程语言中,数组索引是从0开始的,因此,对于一个长度为n的数组,有效的索引范围是0到n-1。如果尝试访问索引为n或更大的元素,就会发生数组越界错误


数组的下标是有范围限制的
数组的下规定是从0开始的,如果数组有n个元素,最后一个元素的下标就是n-1

所以数组的下标如果小于0,或者大于n-1,就是数组越界访问了,超出了数组合法空间的访问。C语言本身是不做数组下标的越界检查,编译器也不一定报错,但是编译器不报错,并不意味着程序就是正确的,

所以程序员写代码时,最好自己做越界的检查

数组越界:

int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int i = 0;for (i = 0; i <= 10; i++){printf("%d\n", arr[i]);//当i等于10的时候,越界访问了}return 0;
}

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


数组越界的影响

  • 程序崩溃: 在许多编程语言中,如C/C++、Java,数组越界通常会导致程序崩溃或抛出异常,因为系统不允许访问未分配的内存区域。
  • 数据损坏: 在某些情况下,如果数组越界没有被系统捕获,可能会覆盖或损坏内存中的其他数据,导致难以跟踪的错误。
  • 安全漏洞: 在安全敏感的应用程序中,数组越界可能被恶意用户利用来执行未授权的操作或访问敏感数据

📝5. 数组作为函数参数

数组作为函数参数在C或C++等编程语言中是一个常见的操作。然而,当数组作为函数参数传递时,它并不直接传递整个数组的内容,而是传递了数组的首地址(即数组第一个元素的地址)。这意味着函数内部对数组的任何非本地(即不在函数内部定义的)修改都会影响到原始数组


⛰️数组作为函数参数错误设计

代码示例 (C语言):

void bubble_sort(int arr[])
{int sz = sizeof(arr) / sizeof(arr[0]);//这样对吗?int i = 0;for (i = 0; i < sz - 1; i++){int j = 0;for (j = 0; j < sz - i - 1; j++){if (arr[j] > arr[j + 1]){int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}
int main()
{int arr[] = { 3,1,7,5,8,9,0,2,4,6 };bubble_sort(arr);//是否可以正常排序?for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}return 0;
}

此处冒泡排序并不能发挥作用
在这里插入图片描述

在这里插入图片描述

注意:数组作为函数参数的时候,并不是把整个数组的传递过去,而是传的是首元素的地址


🏞️数组名

数组名的含义代码示例 (C语言):

int main()
{int arr[10] = { 1,2,3,4,5 };printf("%p\n", arr);printf("%p\n", &arr[0]);printf("%d\n", *arr);//输出结果return 0;
}

在这里插入图片描述

结论: 数组名是数组首元素的地址(有两个例外)

数组名是首元素地址,但是计算大小不适用

int arr[10] = {0};
printf("%d\n", sizeof(arr)); // 40

两个例外:

  • sizeof(数组名),计算整个数组的大小,sizeof内部单独放一个数组名,数组名表示整个数组
  • &数组名,取出的是数组的地址。&数组名,数组名表示整个数组

🌄正确设计

void bubble_sort(int arr[], int sz)//参数接收数组元素个数
{int i = 0;for (i = 0; i < sz - 1; i++){int j = 0;for (j = 0; j < sz - i - 1; j++){if (arr[j] > arr[j + 1]){int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}
int main()
{int arr[] = {3,1,7,5,8,9,0,2,4,6};int sz = sizeof(arr)/sizeof(arr[0]);// 我们先计算好数组的大小bubble_sort(arr, sz);for(int i=0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}

📖6. 总结

在探索C语言数组的旅程即将结束之际,我们不禁要回顾这一路上所见的风景与收获。数组,作为C语言乃至众多编程语言中的基石之一,其重要性不言而喻。它不仅是我们存储和操作一系列相同类型数据的高效工具,更是构建复杂数据结构(如矩阵、字符串等)的基础

通过本文的介绍,我们深入了解了C语言数组的定义、初始化、访问以及通过循环遍历数组的方法。我们见证了数组如何帮助解决一系列实际问题,从简单的数学计算到复杂的数据处理,数组都以其独特的方式展现出了其强大的功能性和灵活性

然而,正如任何强大的工具一样,数组的使用也需要谨慎。越界访问、内存泄漏等问题是我们在使用数组时不得不面对的挑战。因此,掌握良好的编程习惯,如在使用前检查数组边界、合理管理内存等,对于避免潜在的问题至关重要

让我们在掌握C语言函数的基础上,继续保持对编程的热情与好奇心,勇于探索未知,不断挑战自我!!!

在这里插入图片描述
希望本文能够为你提供有益的参考和启示,让我们一起在编程的道路上不断前行!
谢谢大家支持本篇到这里就结束了,祝大家天天开心!

在这里插入图片描述

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

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

相关文章

【C++】学习笔记——AVL树

文章目录 十六、AVL树1. AVL树的概念2. AVL树节点的定义3. AVL树的插入4. AVL树的旋转5. AVL树的验证6. 完整代码测试7. AVL树的性能 未完待续 十六、AVL树 1. AVL树的概念 二叉搜索树虽可以缩短查找的效率&#xff0c;但如果数据有序或接近有序二叉搜索树将退化为单支树&…

前端基础之JavaScript学习——函数的使用

大家好我是来自CSDN的前端寄术区博主PleaSure乐事&#xff0c;今天我们继续有关JavaScript的学习&#xff0c;使用的编译器为vscode&#xff0c;浏览器为谷歌浏览器。 函数的声明与使用 声明 在JavaScript当中函数的声明和其他语言类似&#xff0c;使用如下格式即可声明&…

实战篇(十):使用Processing创建可爱花朵:实现随机位置、大小和颜色的花朵

使用Processing创建可爱花朵 0.效果预览1. 引言2. 设置Processing环境3. 创建花朵类4. 实现花瓣绘制5. 绘制可爱的笑脸6. 鼠标点击生成花朵7. 完整代码8. 总结与扩展0.效果预览 在本教程中,我们将使用Processing编程语言来创建一个可爱的花朵生成器。通过封装花朵为一个类,并…

大语言模型-检索测评指标

1. MRR &#xff08;Mean Reciprocal Rank&#xff09;平均倒数排名&#xff1a; 衡量检索结果排序质量的指标。 计算方式&#xff1a; 对于每个查询&#xff0c;计算被正确检索的文档的最高排名的倒数的平均值&#xff0c;再对所有查询的平均值取均值。 意义&#xff1a; 衡量…

【STM32】按键控制LED光敏传感器控制蜂鸣器(江科大)

一、按键控制LED LED.c #include "stm32f10x.h" // Device header/*** 函 数&#xff1a;LED初始化* 参 数&#xff1a;无* 返 回 值&#xff1a;无*/ void LED_Init(void) {/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENAB…

199.二叉树的右视图(DFS)

给定一个二叉树的根节点 root&#xff0c;想象自己站在它的右侧&#xff0c;按照从顶部到底部的顺序&#xff0c;返回从右侧所能看到的节点值。 示例 1: 输入: [1,2,3,null,5,null,4] 输出: [1,3,4] 示例 2: 输入: [1,null,3] 输出: [1,3] 示例 3: 输入: [] 输出: [] 解题…

贪心算法总结(1)

一、贪心算法简介 常用方法&#xff1a;交换论证法、数学归纳法、反证法、分类讨论 二、柠檬水找零&#xff08;交换论证法&#xff09; . - 力扣&#xff08;LeetCode&#xff09; class Solution { public:bool lemonadeChange(vector<int>& bills) {int five0,t…

【考研数学】线代满分经验分享+备考复盘

我一战二战复习都听了李永乐的线代课&#xff0c;二战的时候只听了一遍强化&#xff0c;个人感觉没有很乱&#xff0c;永乐大帝的课逻辑还是很清晰的。 以下是我听向量这一章后根据听课内容和讲义例题总结的部分思维导图&#xff0c;永乐大帝讲课的时候也会特意点到线代前后联…

TK秘籍:深度剖析机房IP与住宅IP的利与弊

大家好&#xff0c;今天我们来聊聊TikTok运营中的一个重要环节——IP地址的选择。 想象一下&#xff0c;你在TikTok上发布视频&#xff0c;就像是在一个热闹的市集上摆摊&#xff0c;而IP地址就是你的摊位位置。选对了位置&#xff0c;你的摊位就能吸引更多顾客&#xff0c;也…

最小二乘求待定位点的位置(三维环境)|MATLAB

前言 之前发过三点法求待测点位置的程序讲解&#xff0c;哪个是二维的&#xff0c;见&#xff1a;基于伪逆的三点法距离求位置&#xff0c;MATLAB源代码&#xff08;MATLAB函数&#xff09; 这里给出三维情况下的函数和测试代码。对于函数&#xff0c;输入已知锚点的位置、待…

JavaEE:Spring Web简单小项目实践三(留言板实现)

学习目的&#xff1a; 1、理解前后端交互过程 2、学习接口传参&#xff0c;数据返回以及页面展示 目录 1、准备工作 2、约定前后端交互接口 1、获取全部留言 2、发表新留言 3、实现服务器端代码 4、调整前端页面代码 5、运行测试 1、准备工作 创建SpringBoot项目&#x…

Linux 服务器管理和维护

Linux 是一个非常严谨的操作系统&#xff0c;每个目录都有自己的作用&#xff0c;这些作用是固定的&#xff0c;没有特殊情况&#xff0c;应严格执行&#xff1b; Linux 中所有东西以文件形式存储和管理&#xff0c;命令也不例外&#xff1b; 以下四个 bin 是二进制文件&…

SVM 技能测试:25 个 MCQ 用于测试数据科学家的 SVM

SVM 技能测试:25 个 MCQ 用于测试数据科学家的 SVM(2024 年更新) 一、介绍 你可以把机器学习算法想象成一个装满斧头、剑和刀片的军械库。你有各种各样的工具,但你应该学会在正确的时间使用它们。打个比方,将“线性回归或逻辑回归”视为一把能够有效地切片和切块数据但…

LeetCode 739, 82, 106

文章目录 739. 每日温度题目链接标签思路代码 82. 删除排序链表中的重复元素 II题目链接标签思路代码 106. 从中序与后序遍历序列构造二叉树题目链接标签思路二叉树的三种遍历值与索引的映射对于后序遍历的使用对于中序遍历的使用 代码 739. 每日温度 题目链接 739. 每日温度…

jenkins 插件版本冲突

一、Jenkins安装git parameter 插件重启后报错与临时解决方案 cd /root/.jenkins cp config.xml config.xml.bak vim config.xml <authorizationStrategy class"hudson.security.FullControlOnceLoggedInAuthorizationStrategy"><denyAnonymousReadAcces…

【工具使用】EMACS的verilog_mode脚本

#工作记录# 俗话说不会玩连连看的工程师不是一个好的SoC工程师。 在做集成工作的时候&#xff0c;集成连线估计是一件比较繁琐且容易出错的事情&#xff0c;连线类型定义出错、位宽问题、连线众多等等问题&#xff0c;此时使用由Veripool带来的verilog_mode简直是令人神清气爽…

基于牛顿-拉夫逊优化算法(Newton-Raphson-based optimizer, NBRO)的无人机三维路径规划

牛顿-拉夫逊优化算法(Newton-Raphson-based optimizer, NBRO)是一种新型的元启发式算法&#xff08;智能优化算法&#xff09;&#xff0c;该成果由Sowmya等人于2024年2月发表在中科院2区Top SCI期刊《Engineering Applications of Artificial Intelligence》上。 1、算法原理…

制造运营管理系统(MOM系统),企业实现先进制造的关键一步

随着全球制造业的快速发展&#xff0c;企业对于生产效率和成本控制的要求日益增高。在这个背景下&#xff0c;制造运营管理系统&#xff08;MOM系统&#xff09;成为了企业提升竞争力的关键工具。盘古信息作为业内领先的智能制造解决方案提供商&#xff0c;其MOM系统更是以其卓…

首批通过 | 百度通过中国信通院H5端人脸识别安全能力评估工作

2024年5月&#xff0c;中国信息通信研究院人工智能研究所依托中国人工智能产业发展联盟安全治理委员会&#xff08;AIIA&#xff09;、“可信人脸应用守护计划”及多家企业代表共同开展《H5端人脸识别线上身份认证安全能力要求及评估方法》的编制工作&#xff0c;并基于该方法开…

COD论文笔记 Deep Gradient Learning for Efficient Camouflaged 2022

动机 这篇论文的动机在于解决伪装目标检测(COD)中的一个关键问题&#xff1a;在复杂背景下&#xff0c;伪装目标与背景的边界模糊&#xff0c;使得检测变得极其困难。现有的方法&#xff0c;如基于边界或不确定性的模型&#xff0c;通常仅响应于伪装目标的稀疏边缘&#xff0c…