C语言【数组】从入门到精通

简介: C语言【数组】从入门到精通

🐷数组


C 语言支持数组数据结构,它可以存储一个固定大小的相同类型元素的顺序集合。数组是用来存储一系列数据,但它往往被认为是一系列相同类型的变量。

数组的声明并不是声明一个个单独的变量,比如 runoob0、runoob1、…、runoob99,而是声明一个数组变量,比如 runoob,然后使用 runoob[0]、runoob[1]、…、runoob[99] 来代表一个个单独的变量。

所有的数组都是由连续的内存位置组成。最低的地址对应第一个元素,最高的地址对应最后一个元素。



数组中的特定元素可以通过索引访问,第一个索引值为 0。




🐸1.一维数组的创建与初始化


🐺数组的创建:


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


格式:type_t arr_name [const_n]


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


int main() 
{
  //创建一个数组-存放十个整型-数据
  int arr[10] = {1,2,3,5};//不完全初始化,剩下的元素默认初始化为0
  char ch[5] = {'a','b'};
  char ch1[5] = "ab";
  char ch2[] = "abcdef";//未指定数组大小,必须进行初始化,他会根据初始化内容给数组指定大小
  printf("%d\n", sizeof(ch2));//7
  //size 计算ch2所占空间的大小
  printf("%d\n", strlen(ch2));//6
  //strlen 求字符串长度 遇到\0停止  ,且\0不计入字符串长度
  return 0
  }


注:


1.strlen和sizeof没有什么关联

2.strlen 是求字符串长度的-只能针对于字符长长度 - 库函数 -使用得引头文件

3.sizeof 计算变量,数组,类型的大小 - 单位是字节 - 操作符


初始化 时:大括号 { } 之间的值的数目不能大于我们在数组声明时在方括号 [ ] 中指定的元素数目。如果您省略掉了数组的大小,数组的大小则为初始化时元素的个数。

🐶2.一维数组的使用


对于数组的使用,操作符: [ ] ,下标引用操作符,间接访问数组的操作符。


🦮一维数组的定义


一维数组的定义方式

在C语言中使用数组必须先进行定义。一维数组的定义方式为:


类型说明符 数组名 [常量表达式];


其中,类型说明符是任一种基本数据类型或构造数据类型。数组名是用户定义的数组标识符。方括号中的常量表达式表示数据元素的个数,也称为数组的长度。例如:


int a[10];  /* 说明整型数组a,有10个元素 */
float b[10], c[20];  /* 说明实型数组b,有10个元素,实型数组c,有20个元素 */
char ch[20];  /* 说明字符数组ch,有20个元素 */


对于数组类型说明应注意以下几点:


数组的类型实际上是指数组元素的取值类型。对于同一个数组,其所有元素的数据类型都是相同的。

数组名的书写规则应符合标识符的书写规定。

数组名不能与其它变量名相同。

方括号中常量表达式表示数组元素的个数,如a[5]表示数组a有5个元素。但是其下标从0开始计算。因此5个元素分别为a[0], a[1], a[2], a[3], a[4]。

不能在方括号中用变量来表示元素的个数,但是可以是符号常数或常量表达式。

允许在同一个类型说明中,说明多个数组和多个变量。


🐟一维数组的引用


数组元素是组成数组的基本单元。数组元素也是一种变量, 其标识方法为数组名后跟一个下标。下标表示了元素在数组中的顺序号。数组元素的一般形式为:

数组名[下标]

其中下标只能为整型常量或整型表达式。


数组元素通常也称为下标变量。必须先定义数组,才能使用下标变量。在C语言中只能逐个地使用下标变量,而不能一次引用整个数组。


打印数组,下标访问:



总结:


1.数组是使用下表来访问的,下标是从0开始的

2. 数组的大小可以通过计算得到


int sz = sizeof(arr)/sizeof(arr[0]);


🦖3.一维数组在内存中的储存


先看代码:


int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,0};
  int sz = sizeof(arr)/sizeof(arr[0]);
  for (int i = 0; i < sz; i++) {
  printf("&arr[%d]=%p\n",i, &arr[i]);//打印地址用%p
  }
  return 0;
}

上面的数据类型是int型,一个数据在内存中开辟4个字节,又因为16进制是由1,2,3,4,5,6,7,8,9,a,b,c,d,e,f组成

由此不难看出,随着数组下标的增长,元素的地址,也在有规律的变化,因此得出==数组在内存中是连续存放的==


🐇4.二维数组的创建


数组创建:

二维数组定义的一般形式是:


dataType arrayName [length1][length2];


其中,dataType 为数据类型,arrayName 为数组名,length1 为第一维下标的长度,length2 为第二维下标的长度。

我们可以将二维数组看做一个 Excel 表格,有行有列,length1 表示行数,length2 表示列数,要在二维数组中定位某个元素,必须同时指明行和列。


 

int arr1[3][4];//三行四列
  char arr2[3][5];
  double arr3[3][4];


例,arr1定义了一个 3 行 4 列的二维数组,共有 3×4=12 个元素,数组名为 arr1,即:


a[0][0], a[0][1], a[0][2], a[0][3]

a[1][0], a[1][1], a[1][2], a[1][3]

a[2][0], a[2][1], a[2][2], a[2][3]


在C语言中,二维数组是按行排列的。也就是先存放 a[0] 行,再存放 a[1] 行,最后存放 a[2] 行;每行中的 4 个元素也是依次存放。数组 arr1 为 int 类型,每个元素占用 4 个字节,整个数组共占用 4×(3×4)=48 个字节。

🐔5.二维数组的初始化


二维数组的初始化可以按行分段赋值,也可按行连续赋值。


int arr4[3][4] = { {1,2,3},{4,5,6},{7,8,9},{10,11,12} };
int arr5[3][4]={1,2,3,4,5,6,7,8,9,10,11,12}


这两种赋初值的结果是完全相同的。


对于二维数组的初始化还要注意以下几点:


可以只对部分元素赋值,未赋值的元素自动取“零”值。

如果对全部元素赋值,那么第一维的长度可以不给出。

二维数组可以看作是由一维数组嵌套而成的;如果一个数组的每个元素又是一个数组,那么它就是二维数组。当然,前提是各个元素的类型必须相同。


🦀6.二维数组的使用


二维数组依旧是利用下标访问!!!


int arr4[3][4] = { {1,2,3},{4,5,} };
  //  1  2  3  0
  //  4  5  0  0
  //  0  0  0  0
  int i = 0;
  for (i = 0; i < 3; i++) {
    int j = 0;
    for (j = 0; j < 4; j++) {
    printf("%d", arr4[i][j]);
    }
    printf("\n");
  }



打印地址:



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



二维数组在逻辑(表现形式)上可理解为矩阵形式(分行分列),但其物理存储形式却是连续的,即存完第一行,在其后面接着存储第二行,第三行,…,如无特殊说明,本书中涉及对二维数组的表述一般指的是其逻辑形式即矩阵形式。



🙈7.数组作为函数参数


数组用作函数参数有两种形式,一种是把数组元素(下标变量)作为实参使用;另一种是把数组名作为函数的形参和实参使用。 数组元素就是下标变量,它与普通变量并无区别。 因此它作为函数实参使用与普通变量是完全相同的,在发生函数调用时,把作为实参的数组元素的值传送给形参,实现单向的值传送。


🦄数组元素作函数实参


数组元素就是下标变量,它与普通变量并无区别。 因此它作为函数实参使用与普通变量是完全相同的,在发生函数调用时,把作为实参的数组元素的值传送给形参,实现单向的值传送。


🐧数组名作为函数参数


用数组名作函数参数与用数组元素作实参有几点不同。


用数组元素作实参时,只要数组类型和函数的形参变量的类型一致,那么作为下标变量的数组元素的类型也和函数形参变量的类型是一致的。因此,并不要求函数的形参也是下标变量。换句话说,对数组元素的处理是按普通变量对待的。用数组名作函数参数时,则要求形参和相对应的实参都必须是类型相同的数组,都必须有明确的数组说明。当形参和实参二者不一致时,即会发生错误。

在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。在函数调用时发生的值传送是把实参变量的值赋予形参变量。在用数组名作函数参数时,不是进行值的传送,即不是把实参数组的每一个元素的值都赋予形参数组的各个元素。因为实际上形参数组并不存在,编译系统不为形参数组分配内存。那么,数据的传送是如何实现的呢?在我们曾介绍过,数组名就是数组的首地址。因此在数组名作函数参数时所进行的传送只是地址的传送,也就是说把实参数组的首地址赋予形参数组名。形参数组名取得该首地址之后,也就等于有了实在的数组。实际上是形参数组和实参数组为同一数组,共同拥有一段内存空间。



上图说明了这种情形。图中设a为实参数组,类型为整型。a占有以2000为首地址的一块内存区。b为形参数组名。当发生函数调用时,进行地址传送,把实参数组a的首地址传送给形参数组名b,于是b也取得该地址2000。于是a,b两数组共同占有以2000为首地址的一段连续内存单元。从图中还可以看出a和b下标相同的元素实际上也占相同的两个内存单元(整型数组每个元素占二字节)。例如a[0]和b[0]都占用2000和2001单元,当然a[0]等于b[0]。类推则有a[i]等于b[i]。

目录
相关文章
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
86 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
传感器 算法 安全
【C语言】两个数组比较详解
比较两个数组在C语言中有多种实现方法,选择合适的方法取决于具体的应用场景和性能要求。从逐元素比较到使用`memcmp`函数,再到指针优化,每种方法都有其优点和适用范围。在嵌入式系统中,考虑性能和资源限制尤为重要。通过合理选择和优化,可以有效提高程序的运行效率和可靠性。
101 6
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
73 5
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
61 4
|
3月前
|
存储 编译器 C语言
【c语言】数组
本文介绍了数组的基本概念及一维和二维数组的创建、初始化、使用方法及其在内存中的存储形式。一维数组通过下标访问元素,支持初始化和动态输入输出。二维数组则通过行和列的下标访问元素,同样支持初始化和动态输入输出。此外,还简要介绍了C99标准中的变长数组,允许在运行时根据变量创建数组,但不能初始化。
65 6
|
3月前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
3月前
|
存储 C语言
C语言:一维数组的不初始化、部分初始化、完全初始化的不同点
C语言中一维数组的初始化有三种情况:不初始化时,数组元素的值是随机的;部分初始化时,未指定的元素会被自动赋值为0;完全初始化时,所有元素都被赋予了初始值。