数组的本质:
在 C 语言中,数组是同类型数据的集合,它在内存中占用一块连续的空间。数组的大小是固定的,并且数组的元素是按顺序存储的。数组的首元素的地址就是数组名,数组名相当于一个指针,指向数组的第一个元素。
数组的定义:
int arr[5];
这段代码定义了一个包含5个整数的数组。数组 arr
在内存中占用的是一块连续的空间,数组名 arr
实际上代表了数组首元素的地址。
指针与数组:
指针是一个存储内存地址的变量。指针本身可以指向任何类型的数据,比如指向整数、字符、结构体等。当我们定义一个指针时,指针变量需要存储一个地址,并且这个地址必须指向相应类型的数据。
指针的定义:
int* p;
p = arr;
这里,指针 p
指向了数组 arr
的首元素。此时,指针 p
和数组 arr
有着密切的关系,它们指向的内存地址是相同的。
在 C 语言中,数组和指针是可以互换使用的。这是由于数组名代表的是指向数组首元素的指针。事实上,数组名和指针的关系让我们可以使用指针访问数组中的元素。具体来说,数组下标运算符([]
)和指针运算符(*
)有着紧密的联系。
通过指针可以访问数组的元素,这是 C 语言中的一个重要特性。实际上,数组下标运算符和指针运算符在本质上是等价的,下面是几种等价的表达方式:
arr[i]
和 *(arr + i)
arr[i]
是数组下标的常见形式,访问数组中第 i
个元素。*(arr + i)
是通过指针运算的方式访问数组中第 i
个元素。这里,arr
是数组名,表示数组首元素的地址,arr + i
就是数组首元素地址偏移 i
个位置,*
用于解引用得到相应的值。*(arr + i)
和 *(i + arr)
arr + i
和 i + arr
都指向数组中第 i
个元素。通过 *
运算符解引用后,它们的结果相同。arr[i]
和 i[arr]
i[arr]
看起来不太直观,但它与 arr[i]
是完全等价的。C 语言中的数组访问本质上是通过指针加偏移量来完成的,因此 arr[i]
实际上是 *(arr + i)
,而 i[arr]
是 *(i + arr)
,它们的运算结果是一样的。在 C 语言中,数组和指针的关系使得某些看起来不同的写法,实际上是完全等价的。以下是六个等价的表达式:
arr[i]
和 *(arr + i)
arr[i]
是常见的数组下标访问方式,而 *(arr + i)
则是通过指针访问第 i
个元素。两者等价。*(arr + i)
和 *(i + arr)
*(arr + i)
和 *(i + arr)
也是等价的,指向的是同一元素。arr[i]
和 i[arr]
arr[i]
和 i[arr]
是等价的表达方式,虽然后者不常见,但在 C 语言中合法且结果相同。*(arr + i)
和 p[i]
p[i]
是指针访问数组的常见形式,等同于 *(p + i)
,通过偏移量访问数组中的元素。*(arr + i)
和 *(p + i)
*(arr + i)
和 *(p + i)
之间的关系类似,都是通过指针加偏移量来访问数组元素。*(i + arr)
和 i[arr]
*(i + arr)
和 i[arr]
指向的是相同的内存位置,因而是等价的。尽管 i[arr]
是合法的且等价于 arr[i]
,但是这种写法通常不推荐使用,因为它不直观,可能让代码的阅读者产生困惑。通常我们更倾向于使用 arr[i]
或 p[i]
这样的常见形式,这样代码更加清晰易懂。
我们来分析一个简单的程序,演示如何使用指针访问数组。
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdio.h>
int main() {
int arr[10] = { 0 }; // 定义一个大小为 10 的整数数组
int* p = arr; // 定义指针 p,指向数组 arr
int sz = sizeof(arr) / sizeof(arr[0]); // 计算数组元素个数
int i = 0;
// 输入 10 个整数
for (i = 0; i < sz; i++) {
scanf("%d", p + i); // 使用指针来访问数组
}
printf("使用arr[i]:");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", arr[i]); // 使用指针访问数组并输出
}
printf("\n");
printf("使用*(arr + i):");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", *(arr + i)); // 使用指针访问数组并输出
}
printf("\n");
printf("使用*(i + arr):");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", *(i + arr)); // 使用指针访问数组并输出
}
printf("\n");
printf("使用arr[i]:");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", arr[i]); // 使用指针访问数组并输出
}
printf("\n");
printf("使用i[p]:");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", i[p]); // 使用指针访问数组并输出
}
printf("\n");
printf("使用p[i]:");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", p[i]); // 使用指针访问数组并输出
}
printf("\n");
printf("使用*(p + i):");
// 输出 10 个整数
for (i = 0; i < sz; i++) {
printf("%d ", *(p + i)); // 使用指针访问数组并输出
}
return 0;
}
这段代码展示了如何在 C 语言中使用数组和指针的不同方式来访问和输出数组的元素。通过指针运算,可以以多种形式访问和输出数组的内容。让我们逐步分析这段代码。
int arr[10] = { 0 }; // 定义一个大小为 10 的整数数组
int* p = arr; // 定义指针 p,指向数组 arr
int sz = sizeof(arr) / sizeof(arr[0]); // 计算数组元素个数
arr
是一个大小为 10 的整型数组,初始化为 0
。p
是一个指针,指向 arr
数组的首元素。sz
是数组 arr
的大小,通过 sizeof(arr) / sizeof(arr[0])
来计算数组的元素个数。这里 sizeof(arr)
获取数组总大小,sizeof(arr[0])
获取数组单个元素的大小,因此 sz
即为数组中元素的个数。for (i = 0; i < sz; i++) {
scanf("%d", p + i); // 使用指针来访问数组
}
scanf
输入 10 个整数,并将它们存入数组 arr
中。这里通过 p + i
来操作数组,因为 p
是指向数组首元素的指针,p + i
相当于指向数组第 i
个元素的指针,scanf
会把输入的整数存入该位置。接下来的代码展示了如何用不同的方式输出数组的元素:
arr[i]
输出数组元素printf("使用arr[i]:");
for (i = 0; i < sz; i++) {
printf("%d ", arr[i]); // 直接使用数组下标访问数组元素并输出
}
arr[i]
直接访问数组元素并输出。这是最常见的访问数组元素的方式。*(arr + i)
输出数组元素printf("使用*(arr + i):");
for (i = 0; i < sz; i++) {
printf("%d ", *(arr + i)); // 使用指针运算来访问数组元素并输出
}
*(arr + i)
是通过指针加偏移量来访问数组元素。arr
是数组首元素的地址,arr + i
就是数组第 i
个元素的地址,*
用来解引用,得到该地址存储的值。*(i + arr)
输出数组元素printf("使用*(i + arr):");
for (i = 0; i < sz; i++) {
printf("%d ", *(i + arr)); // 使用指针运算来访问数组元素并输出
}
*(i + arr)
和 *(arr + i)
完全等价,它们的运算方式是一样的。这里也展示了指针偏移量的交换性。i[p]
输出数组元素printf("使用i[p]:");
for (i = 0; i < sz; i++) {
printf("%d ", i[p]); // 通过指针加偏移量来访问数组元素并输出
}
i[p]
是合法且等价于 p[i]
的写法。它看起来不太直观,但本质上是通过指针偏移量访问数组元素。实际上,i[p]
是等同于 *(i + p)
,和 *(p + i)
相同。p[i]
输出数组元素printf("使用p[i]:");
for (i = 0; i < sz; i++) {
printf("%d ", p[i]); // 使用指针访问数组并输出
}
p[i]
是通过指针访问数组元素的常见方式,等同于 *(p + i)
。*(p + i)
输出数组元素printf("使用*(p + i):");
for (i = 0; i < sz; i++) {
printf("%d ", *(p + i)); // 使用指针运算来访问数组元素并输出
}
*(p + i)
是通过指针偏移量访问数组元素。p
是指向数组首元素的指针,p + i
就是数组第 i
个元素的指针,*
解引用该指针并输出元素值。假设你输入了以下数据:
1 2 3 4 5 6 7 8 9 10
输出结果将是:
使用arr[i]:1 2 3 4 5 6 7 8 9 10
使用*(arr + i):1 2 3 4 5 6 7 8 9 10
使用*(i + arr):1 2 3 4 5 6 7 8 9 10
使用arr[i]:1 2 3 4 5 6 7 8 9 10
使用i[p]:1 2 3 4 5 6 7 8 9 10
使用p[i]:1 2 3 4 5 6 7 8 9 10
使用*(p + i):1 2 3 4 5 6 7 8 9 10
这段代码展示了数组和指针之间的密切关系。通过不同的指针访问方式,如 arr[i]
、*(arr + i)
、*(i + arr)
、i[p]
和 p[i]
,我们可以看到这些方式实际上是等价的,它们都可以用来访问数组中的元素。通过指针的运算,可以非常灵活地操作数组元素。
本文详细分析了 C 语言中的数组与指针,特别是它们之间的等价表达方式。我们发现,数组下标运算符和指针运算符在很多情况下是等价的,指针的使用不仅可以访问数组,还能使得程序更加灵活与高效。通过理解数组名、指针和下标之间的关系,程序员可以在编写代码时更加得心应手。
对于数组和指针的相关表达式,尽管它们在语法上可能有不同的形式,但它们在底层的运作原理是相同的。掌握这些等价的表达方式,将有助于你在 C 语言编程中更加灵活地运用数组与指针。