自学内容网 自学内容网

C语言--数组

        数组是 C 语言中最基础却最强大的数据结构之一。无论是在简单的日常编程中,还是在复杂的工程实践中,数组都扮演着重要角色。这篇博客从数组的基础概念出发,逐步探讨一维数组和多维数组的创建、初始化、使用及内存管理,并补充一些实用的例子和练习,帮助读者全面掌握数组的使用。

1. 数组的基本概念

什么是数组?

数组是存储相同数据类型的元素集合。它的特点如下:

  1. 存储多个数据:数组用于保存多个元素,元素数量必须大于0。

  2. 类型相同:数组中的所有元素类型必须一致。

数组分为一维数组和多维数组,最常见的是一维和二维数组。

2. 一维数组

2.1 创建一维数组

创建数组时需要指定类型、数组名和数组大小。例如:

int math[20]; // 存储某班级20人的数学成绩
char ch[8];   // 存储8个字符

2.2 初始化一维数组

在创建数组时,常用大括号 {} 给定初始值:

int arr[5] = {1, 2, 3, 4, 5};  // 完全初始化
int arr2[6] = {1};             // 部分初始化,未赋值部分默认为0

 注意:初始化的值不可以比数组的元素多,否则是错误初始化。

2.3 数组类型

数组也是一种自定义类型。去掉数组名留下的就是数组类型。例如:

int arr[10]; // 类型为 int[10]

注意:int[5]和int[10],不是一种类型。并且其中的int不是数组类型,是数组中元素的类型。

3. 使用一维数组

3.1 访问数组元素

通过下标访问数组,C 语言规定下标从 0 开始,所以假设数组有n个元素,最后一个元素,最后一个元素的下标就是n-1,下表相当于数组元素的编号。例如:

int arr[5] = { 1, 2, 3, 4, 5 };
printf("%d\n", arr[4]);//输出第5个元素:5

3.2 遍历数组

使用 for 循环打印所有元素:

for (int i = 0; i < 5; i++) 
{
    printf("%d ", arr[i]);
}
//输出1 2 3 4 5

3.3 动态输入数组元素

用户可以通过输入改变数组内容:

for (int i = 0; i < 10; i++) 
{
    scanf("%d", &arr[i]);
}
//输出的数字等于输入的数字

补充:对于数组只能一个一个输入、一个一个输出<字符数组除外> 。

          其中输入和输出都推荐使用循环来解决。

4. 一维数组的内存存储

数组在内存中是连续存储的,内存被划分为一个个的内存单元,一个单元的大小是一个字节,每个内存单元,都有一个编号,编号就是地址,元素地址之间的差值取决于元素类型的大小。例如,整型数组中,每个元素占4个字节,相邻地址之间相差4。

通过打印数组的地址可以验证这一点:

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

输出结果: 

 补充:1. 可以观察到每一个数组元素之间的地址间隔是4,这是由于数组元素的数据类型决定的。

            2. 上面的输出结果过于复杂,可以将编译环境由x64(64位)改成x86(32位)32位地址较短便于观察。

            3. 其中可以得出结论数组在内存中随着下标的增长,地址是由大变小的,存储方式是连续的,这也为后续利用指针访问数组打下了基础。

5. 使用 sizeof 计算数组元素个数

通过 sizeof 关键字,可以计算数组所占的总字节数和单个元素的字节数(单位是字节),从而得出元素个数:

#include <stdio.h>
int main()
{
int arr[5] = { 1, 2, 3, 4, 5 };
printf("%d\n", sizeof(arr));
return 0;
}//输出结果是20

计算结果是数组所占内存空间的大小,单位是字节。又因为数组里面的元素数据类型是一样的,所以只要计算出一个元素所占字节的的个数,数组的元素个数就能计算出来:

#include <stdio.h>
 
int main()
{
int arr[5] = { 1, 2, 3, 4, 5 };
int sz = sizeof(arr) / sizeof(arr[0]);
printf("%d", sz);
return 0;
}//这里的结果是5,表示数组中有5个元素

 补充:所以以后在写代码时,就不用写的那么死板,使用上面的程序,无论数组怎么变化都可以得到数组的大小。

6. 二维数组

二维数组是以一维数组为基础的一种扩展形式,常用于表示矩阵。

补充:二维数组作为数组元素的数组叫做三维数组,二维数组以上的统称为多维数组。

6.1 创建二维数组

二维数组的基本语法如下:

int arr[3][4];  // 3行4列
  • 3表示数组有3行。

  • 4表示数组有4列。

  • int表示数组的每个元素是整型。

  • arr是数组名,可以根据自己的需要更改。  

  • 数组元素可以通过 arr[行号][列号] 访问。

6.2 初始化二维数组

二维数组的初始化与一维数组类似,可以采用以下方式:

  1. 完全初始化

    int arr[3][5] = {{1, 2, 3, 4, 5}, {2, 3, 4, 5, 6},{3, 4, 5, 6, 7}};
  2. 部分初始化

    int arr[3][5] = { 1, 2 };

  3. 按行初始化:

    int arr[3][5] = { {2, 3},{3, 4},{4, 5} };
    

  4. 初始化是省略行,但是不能省略列:

    int arr1[][5] = { 1, 2 , 3};
    int arr2[][5] = {1, 2, 3, 4, 5, 6, 7};
    int arr3[][5] = { {1, 2},{3, 4},{5, 6} ,{7, 8} };
    

7. 二维数组的使用

7.1 访问二维数组

二维数组也是利用下标的形式访问,只要锁定了行和列就能确定唯一锁定数组中的一个元素(二维数组的行和列也是从0开始的),通过双重下标定位具体元素:

#include <stdio.h>
int main()
{
    int arr[2][3] = {{1, 2, 3}, {4, 5, 6}};
    printf("%d\n", arr[1][2]); 
) // 输出第2行第3列的值:6

7.2 遍历二维数组

通过嵌套循环遍历整个二维数组:

#include <stdio.h>
int main()
{
int arr[3][5] = { {1, 2, 3, 4, 5}, {2, 3, 4, 5, 6},{3, 4, 5, 6, 7} };//创建数组
int i = 0;//创建行的变量
for (i = 0; i < 3; i++)//遍历行
{
int j = 0;//创建列的变量
for (j = 0; j < 5; j++)//遍历列
{
scanf("%d", &arr[i][j]);//依次输入进入二维数组
}
}

for (i = 0; i < 3; i++)//产生行号
{
int j = 0;//产生列号
for (j = 0; j < 5; j++)//遍历数组
{
printf("%d", arr[i][j]);//打印数据
}
printf("\n");

}
return 0;
}
//注意scanf函数如果需要一个数字一个数字的输入需要一个数字一个空格的格式输入,要不然就会因为数据连在一起默认为一个数字

8. 二维数组在内存中的存储

如果想研究二维数组的存储,可以像研究一维数组一样,打印出二维数组每个元素的地址。

#include <stdio.h>
int main()
{
int arr[3][5] = { 0 };
int i = 0;
int j = 0;
for (i = 0; i < 3; i++)
{
for (j = 0; j < 5; j++)
{
printf("&arr[%d][%d] = %p\n", i, j, &arr[i][j]);
}
}
return 0;
}

为了便于观察,还是要求配置x86的环境下编译代码。

从输出结果来看每一行内部的每一个元素都是相邻的,地址之间相差4搁字节,所以在二维数组中每个元素都是连续存放的。

对于二维数组的访问方式(二维数组是一维数组的数组):

 arr[0][0]         arr[0][1]         arr[0][2]         arr[0][3]         arr[0][4]        ....        第一行

 arr[0][j]        .......        j = 0~4

 arr[1][0]         arr[1][1]         arr[1][2]         arr[1][3]         arr[1][4]        ....        第二行

 arr[1][j]        .......        j = 0~4

可以理解为arr[0]是第一行数组的数组名,arr[1]是第二行数组的数组名。

9.C99 中的变长数组 (VLA)

c99标准之前,C语言在创建数组的时候,数组大小的指定只能是常量、常量表达式,如果初始化数据的话可以省略数组的大小。

int arr1[10];            //数组大小由常量表示
int arr2[3+5];           //数组大小由常量表达式表示
int arr3[] = {1,2,3};    //初始化数据数组大小省略

C99 标准引入了 变长数组(Variable Length Array, VLA) 的概念,为数组的定义带来了更高的灵活性。在传统 C 语言中,数组的大小必须是编译时的常量,而 VLA 允许在运行时动态指定数组大小,从而让程序更加灵活高效。简单来说就是允许我们使用变量指定数组大小。

int n = a + b;
int arr[n];  // n 是变量,数组大小在运行时确定

上面的示例程序中,arr就是变长数组他的长度取决于n,而n的值编译器无法事先预知,只有运行的时候才知道是多少。

变长数组的根本特征,就是数组长度只有运行时才能确定,所以变长数组不能初始化。它的好处是程序员不必在开发时,随意为数组指定一个估计的长度,程序可以在运行时为数组分配精确的长度。有一个比较迷惑的点,变长数组的意思是数组的大小是可以使用变量来指定的,在程序运行的时候,根据变量的大小来指定数组的元素个数,而不是说数组的大小是可变的。数组的大小一旦确定就不能再变化了。

但是变长数组也有一些限制,如变长数组存储在栈上,因此如果数组太大,可能导致栈溢出。不支持部分 C99 实现:某些编译器(如 Visual Studio)不支持变长数组等。

下面是利用gcc编译器编写一个有关变长数组的代码:

#include <stdio.h>

int main()
{
int n = 0;
scanf("%d", &n);//根据输入数据确定数组大小
int arr[n];
int i = 0;
for (i = 0; i < n; i++)
{
scanf("%d", &arr[i]);
}
for (i = 0; i < n; i++)
{
printf("%d ",arr[i]);
}
return 0;
}

10.数组的练习

练习1:多个字符从两端移动,向中间汇聚
编写代码,演示多个字符从两端移动,向中间汇聚


原文地址:https://blog.csdn.net/Tanecious/article/details/143981320

免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!