【C语言 | 数组】C语言数组详解(经典,超详细)

😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀
🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C++、数据结构、音视频🍭
🤣本文内容🤣:🍭介绍C语言的数组🍭
😎金句分享😎:🍭🍭

本文未经允许,不得转发!!!

目录

  • 🎄一、了解数组,清楚这五个方面
    • ✨1.1 数组所在的内存地址
    • ✨1.2 数组的内容
    • ✨1.3 数组的类型
    • ✨1.4 数组元素的类型
    • ✨1.5 数组元素的个数
  • 🎄二、数组的几个地址—— a、&a、&a[0]
  • 🎄三、不指定数组长度——int a[]={1,2};
  • 🎄四、数组初始化
  • 🎄五、字符串
  • 🎄六、总结


在这里插入图片描述

🎄一、了解数组,清楚这五个方面

如果定义一个基本数据类型变量,我们可以了解到这三个方面的内容:1、变量所在的内存地址; 2、变量的值; 3、变量的类型;

例如:从语句 int i = 1; ,我们可以知道:

  • 1、编译器分配一块4个字节(sizeof(int))的内存,地址是&i
  • 2、这块内存里面存放了值为4的内容;
  • 3、这个变量是 int 类型的。

如果定义的是数组,则我们需要清楚四个方面的内容:

  • 1、数组所在的内存地址;
  • 2、数组的内容;
  • 3、数组的类型;
  • 4、数组的元素类型;
  • 5、数组的元素个数;

✨1.1 数组所在的内存地址

定义了一个任何类型变量,编译器都会为其分配一块内存来存放该变量,起始地址就是 &变量名 ,大小就是该变量的类型的大小。

所以数组也是如此,定义一个数组后,编译器会为该数组分配一块内存。内存大小,就是数组类型的大小。例如:语句int a[5];,编译器会分配一块内存,起始地址可以通过&a来获取,内存大小是20个字节(sizeof(int [5]))。

数组所在的内存地址(数组的地址),就是 &数组名 的值。


✨1.2 数组的内容

数组的内容是指定义数组时,分配的那块内存里存放的东西,与数组元素类型数组元素个数 有关。

数组的内容就是由若干个 固定大小 的内存块,线性排列组成的一个内存块。这里的若干个就是数组元素个数固定大小就是sizeof(数组元素类型)


✨1.3 数组的类型

针对基本数据类型变量 或 指针变量,大部分人都可以轻易指出该变量的类型是什么。

看例子1.3.1:

// array_type.c
char c;		// char 类型
int i;		// int 类型
char *pc;	// char* 类型
int *pi;	// int* 类型

那么,数组的类型是什么?好像以前都没听过这个词汇!!!

仔细观察上面例子,变量的类型都是在定义语句中,把变量名去掉,就得到该变量的类型。同样地,在数组定义语句中把数组名去掉就是数组的类型。

看例子1.3.2:

#include <stdio.h>int main()
{char ca[10];    // 数组类型是 char [10]int ia[5];      // 数组类型是 int [5]char *pca[8];   // 数组类型是 char *[8]int  *aapi[4][5];//数组类型是 int *[4][5]int  (*apai[4])[5];//数组类型是 int (*[4])[5])printf(" sizeof(aapi)=%lu, %lu\n", sizeof(aapi), sizeof(int *[4][5]));printf(" sizeof(apai)=%lu, %lu\n", sizeof(apai), sizeof(int (*[4])[5]));return 0;
}

从这个例子,可以轻易指出数组的类型在数组定义语句中,把数组名去掉,剩下的就是数组的类型。

但是,在例子中,aapiapai 变量的类型可能又把一部分人的CPU干烧了。
这里需要使用一个右左法则来阅读复杂类型:

右左法则

  • 1、从变量名(没变量名的,从最里层的圆括号)开始,先看右边,再看左边;
  • 2、如果右边是() 则是函数,如果是 [] 则是数组。
  • 3、如果遇到[],后面还是[],就先看完右边的[],再看左边。
    例如:int *pi[4][5];,p先跟[4][5]结合,再跟*结合。

变量aapi,先跟右边[4]结合,说明是一个有4个元素数组;再跟[5]结合,说明数组的每个元素都是带有5个元素数组;再跟左边*结合,表示第二维数组的5个元素都是指针;再跟左边int结合,表示指针指向int类型数据。

变量apai,先跟右边[4]结合,说明是一个有4个元素数组;因为()改变优先级,再跟*结合,说明数组的每个元素都是指针;再跟右边[5]结合,表示每个指针指向带有5个元素的数组;再跟左边int结合,表示这些数组都是int类型的。


✨1.4 数组元素的类型

在数组定义语句中,把数组名和后面的[]去掉,剩下的就是数组元素的类型。

看例子1.4.1:

// array_unit_type.c
#include <stdio.h>int main()
{char ca[10];    // 数组元素类型是 char int ia[5];      // 数组元素类型是 int char *pca[8];   // 数组元素类型是 char *int  *aapi[4][5];//数组元素类型是 int *[5]int  (*apai[4])[5];//数组元素类型是 int (*)[5])printf(" sizeof(aapi[0])=%lu, %lu\n", sizeof(aapi[0]), sizeof(int *[5]));printf(" sizeof(apai[0])=%lu, %lu\n", sizeof(apai[0]), sizeof(int (*)[5]));return 0;
}

例子中,前三个类型都比较容易看明白,int *[5]int (*)[5] 分别是什么类型?
int *[5] 类型表示一个数组,每个数组元素都是int *类型。
int (*)[5]类型表示一个指向包含5个int型元素的数组的指针 。

✨1.5 数组元素的个数

C语言中,在定义数组时,都需要明确的给出数组元素个数。

在数组定义语句中,数组名后面的[]中的数字,就是数组元素的个数。

int a[5];为例子,数组个数可以用表达式:sizeof(a) / sizeof(a[0]) 来获得。

在这里插入图片描述

在这里插入图片描述

🎄二、数组的几个地址—— a、&a、&a[0]

当我们定义一个数组 a 时,编译器根据指定的元素个数和元素的类型分配确定大小(元素类型大小*元素个数)的一块内存,并把这块内存的名字命名为 a。名字 a 一旦与这块内存匹配就不能被改变。

a[0]、a[1]等为 a 的元素,但并非元素的名字。数组的每一个元素都是没有名字的。

  • 数组名a:数组名a作为地址使用时,表示数组首个元素的地址(指针),那它所指向的类型就是数组元素类型,加减一个整数就相当于加减(整数*sizeof(元素类型))
    数组名a可以看成一个指针常量,它的值不能被修改,不能单独作为左值使用。
  • &a&a表示对编译器分配好内存的变量a取地址,得到的是整个数组的地址(指针),其所指向的类型是数组的类型,加减一个整数就相当于加减(整数*sizeof(数组的类型))
  • &a[0]a[0]是数组的首个元素,&a[0]是对数组首个元素取地址,得到的是数组首个元素的地址(指针),那它所指向的类型就是数组元素类型,加减一个整数就相当于加减(整数*sizeof(元素类型))

看例子2.1:

// array_addr.c
#include <stdio.h>int main()
{int a[5];// 数组类型:int [5] ; 数组元素类型:intprintf("a=%p a+1=%p %lx\n",a, a+1,  (unsigned long)a+sizeof(int));printf("&a=%p &a+1=%p %lx\n",&a, &a+1,  (unsigned long)&a+sizeof(int [5]));printf("&a[0]=%p &a[0]+1=%p %lx\n",&a[0], &a[0]+1,  (unsigned long)&a[0]+sizeof(int));printf("a=%lu &a=%lu &a[0]=%lu\n",sizeof(a), sizeof(&a), sizeof(&a[0]));return 0;
}

打印结果如下:
在这里插入图片描述

从结果看,可以得出结论:

  • a、&a、&a[0]作为地址(指针)使用时,三个值都是相等的;
  • a、&a、&a[0]的指针类型不一样,&a的指针类型是数组的类型,a、&a[0]的指针类型是数组元素类型,最后进行指针运算时,a、&a[0]的结果是一样的,与&a存在差别;
  • a、&a、&a[0]进行sizeof计算时,sizeof(a)是这个数组大小,其余两个是指针的大小(32位系统为4,64位系统为8)。

在这里插入图片描述

🎄三、不指定数组长度——int a[]={1,2};

在C语言中,可以使用不指定数组长度的方式来定义和初始化数组。这种情况下,编译器会根据提供的初始化值自动推断数组的长度。

例如,代码int a[]={1,2};定义了一个整型数组a,并用初始值1和2进行了初始化。由于未指定数组长度,编译器会根据提供的初始化值计算数组的大小。

在这个例子中,由于提供了两个初始化值,编译器会推断数组长度为2,因此数组a将具有两个元素:a[0]和a[1]。其值分别为:

a[0] = 1
a[1] = 2

通过这种方式,可以方便地定义和初始化具有不同长度的数组,而无需显式指定数组的长度。但是请注意,这种隐式推断数组长度的方式只适用于在声明时进行初始化的静态和自动(非堆)数组。对于动态分配的数组,仍然需要显式指定数组的长度。

总结起来,使用不指定数组长度的方式定义和初始化数组是C语言中的一种常见用法,编译器会根据提供的初始化值来推断数组的长度。
在这里插入图片描述

🎄四、数组初始化

在C语言中,主要有3种常见的方式可以初始化数组。下面列举了其中的几种方式,并提供了相应的示例:

  • 1、在定义数组时,给各个元素初始化。
    int a[5] = {1, 2, 3, 4, 5};
    
  • 2、部分初始化:只为数组的一部分元素提供初始值,剩余元素会被设置为默认值(0)
    int a[5] = {1, 2}; // a[0]和a[1]被初始化为1和2,a[2]、a[3]和a[4]被初始化为0
    
  • 3、不指定数组长度的方式定义和初始化数组。这是C语言中的一种常见用法,编译器会根据提供的初始化值来推断数组的长度。
    int a[]={1,2}; // 初始化结束后,数组a的长度为2
    char str[] = "Hello"; // 字符数组str会被初始化为包含"Hello"字符串的字符序列
    

这些是C语言中常见的数组初始化方式,你可以根据自己的需求选择适合的方式来初始化数组。
看例子4.1:

// array_init.c
#include <stdio.h>int main()
{int i=0;printf("no init:\n");int a[5];for(i=0; i<(sizeof(a)/sizeof(a[0])); i++){printf("%d, ",a[i]);// 没初始化,打印随机值}printf("\n\n");printf("init1: init all unit\n");int a1[5] = {1,2,3,4,5};for(i=0; i<(sizeof(a1)/sizeof(a1[0])); i++){printf("%d, ",a1[i]);}printf("\n\n");printf("init2: init first unit\n");int a2[5] = {5};for(i=0; i<(sizeof(a2)/sizeof(a2[0])); i++){printf("%d, ",a2[i]);}printf("\n\n");printf("init3: Do not specify length\n");int a3[] = {1,2,3,4,5};for(i=0; i<(sizeof(a3)/sizeof(a3[0])); i++){printf("%d, ",a3[i]);}printf("\n\n");printf("init4: Specify some unit\n");int a4[5] = {[1]=2, [3]=4};for(i=0; i<(sizeof(a4)/sizeof(a4[0])); i++){printf("%d, ",a4[i]);}printf("\n\n");printf("init5: Do not specify length, and specify the unit\n");int a5[] = {[1]=2, [3]=4};for(i=0; i<(sizeof(a5)/sizeof(a5[0])); i++){printf("%d, ",a5[i]);}printf("\n");return 0;
}

运行结果:
在这里插入图片描述

在这里插入图片描述

🎄五、字符串

C语言中的字符串,本质为字符数组,编译器自动在结尾加上 ‘\0’ 字符。

字符串字面值可以用来初始化字符数组:char str[]="abc";

字符串字面值存储于程序的全局只读存诸区,内容不可以修改,地址可以看出常量指针,指针类型是const char * const

字符串字面值的长度可以用strlen函数来获取。

// array_str.c
#include <stdio.h>
#include <string.h>int main()
{// 1、字符串字面值的地址、空间大小、字符串长度printf("str_addr=%p str_size=%lu str_len%lu\n", (char*)"str", sizeof("str"), strlen("str"));// 2、字符串字面值给字符数组初始化unsigned char str[100] = "12345";// 3、字符串字面值是 `const char* const` 指针printf("str_1=[%c] str_end=[%c]\n", *("str"+1), *("str"+strlen("str")));return 0;
}

运行结果:
在这里插入图片描述

在这里插入图片描述

🎄六、总结

本文详细地介绍C语言的数组,数组本质上是一段连续的内存空间,了解数组5个重要内容:数组所在的内存地址、数组的内容、数组的类型、数组元素的类型、数组元素的个数;然后介绍数组的几个地址—— a、&a、&a[0];数组的初始化、字符串等。

在这里插入图片描述
如果文章有帮助的话,点赞👍、收藏⭐,支持一波,谢谢 😁😁😁

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

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

相关文章

【LeetCode:307. 区域和检索 - 数组可修改 | 树状数组 or 线段树】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

保姆级教程之SABO-VMD-CNN-SVM的分类诊断,特征可视化

今天出一期基于SABO-VMD-CNN-SVM的分类诊断。 依旧是采用经典的西储大学轴承数据。基本流程如下&#xff1a; 首先是以最小包络熵为适应度函数&#xff0c;采用SABO优化VMD的两个参数。其次对每种状态的数据进行特征向量的求取&#xff0c;并为每组数据打上标签。然后将数据送入…

VN5620以太网测试——DoIP配置

文章目录 前言一、DoIP简介二、Vector Hardware Configuration三、Diagnostics/ISO TP Configuration四、Diagnostic Console五、添加Ethernet Packet Builder前言 CANoe(CAN open environment)VN5620 :是一个紧凑而强大的接口,用于以太网网络的分析、仿真、测试和验证。 V…

Go 理解零值

在 Go 语言中&#xff0c;零值&#xff08;Zero Value&#xff09;是指在声明变量但没有显式赋值的情况下&#xff0c;变量会被自动赋予一个默认值。这个默认值取决于变量的类型&#xff0c;不同类型的变量会有不同的零值。零值是 Go 语言中的一个重要概念&#xff0c;因为它确…

低代码编辑平台后台实现

背景 之前做过一个前端低代码编辑平台&#xff0c;可以实现简单的移动端页面组件拖拽编辑&#xff1a; https://github.com/li-car-fei/react-visual-design 最近基于C的oatpp框架实现了一下后台。使用oatpp框架做web后台开发时&#xff0c;发现按照官方的示例使用的话&#…

设计模式(4)-行为型模式

行为型模式 行为型模式用于描述程序在运行时复杂的流程控制&#xff0c;即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务&#xff0c;它涉及算法与对象间职责的分配。 行为型模式分为类行为模式和对象行为模式&#xff0c;前者采用继承机制来在类间…

基于模拟退火算法的TSP问题建模求解(Python)

基于模拟退火算法的TSP问题建模求解&#xff08;Python&#xff09; 一、模拟退火算法&#xff08;Simulated Annealing Algorithm&#xff0c;SAA&#xff09;工程背景模拟退火算法用于优化问题求解原理 二、旅行商问题&#xff08;Travelling salesman problem&#xff0c;TS…

Linux脚本shell中将Windos格式字符转换为unix

众所周知&#xff0c;windos的文档直接复制到linux服务器上去&#xff0c;是需要进行格式转换的&#xff0c;否则可能出现以下报错&#xff1a; 解决方法&#xff1a; vim 脚本 输入 :set ff ##会显示字符格式 :set ffunix ##转换为unix格式 :wq ##保存退出

蓝眼开源云盘部署全过程(手动安装)

环境概述&#xff1a; 系统-Centos7.4 数据库-MySQL8 云盘系统-Tank4.0.1 前提&#xff1a;操作系统已完成安装&#xff0c;有外部网络。 一.安装数据库 cd到合适的目录进行下载安装操作&#xff0c;期间不要切换出去。 wget https://dev.mysql.com/get/mysql80-community-r…

WPF程序给按钮增加不同状态的图片

首先我们在资源里添加几个图片&#xff0c;Up&#xff0c;Over和Down状态。 然后我们创建一个Style。默认我们的背景设置成Up 然后在Triggers里添加代码&#xff0c;当Property&#xff1a;IsMouseOver为True的时候更换成Over&#xff1b;当Property&#xff1a;IsPressed为Tr…

如何在 Nginx Proxy Manager(NPM)上部署静态网站

前言 众所周知&#xff0c;我们在之前介绍过 Nginx Proxy Manager&#xff08;以下简称 NPM) 这个反向代理的神器&#xff0c;对于一些 Docker 搭建的 Web 项目&#xff0c;NPM 能够很轻松地给他们做反向代理。 然而对于一些静态网站&#xff0c;小伙伴们可能不知道怎么用 NP…

SQL-LABS

less8 and 11-- 12 发现存在注入点 接下来我们会接着用联合查询 和以往的题目不一样没显错位&#xff0c;也就是没有报错的内容&#xff0c;尝试用盲注 布尔型 length&#xff08;&#xff09;返回长度 substr&#xff08;&#xff09;截取字符串&#xff08;语法substr&a…