C++(数组退化为指针与指针算术)
1. 一维数组
#include <iostream>
// 函数声明,形参为一个数组和一个整数
void printArray(int arr[], int size) {
// 遍历数组并打印每个元素
for (int i = 0; i < size; ++i) {
std::cout << arr[i] << " ";
}
}
int main() {
// 定义一个数组并初始化
int arr[5] = {1, 2, 3, 4, 5};
// 调用函数,传递数组和数组大小
printArray(arr, 5);
return 0;
}
1.1 数组退化
在C++中,当我们定义一个函数参数为数组时,比如 void printArray(int arr[], int size)
,数组参数 int arr[]
实际上被解释为指向数组第一个元素的指针,即 int* arr
。这是因为在C++中,函数参数不能真正地是数组,数组参数会自动退化为指向其第一个元素的指针。这种行为称为“数组退化为指针”。因此,在函数 printArray
中,int arr[]
和 int* arr
是等价的。
1.2 指针算术
1.2.1 arr
是指向第一个元素的地址
当你传递数组 arr
给函数 printArray
时,传递的实际上是指向数组第一个元素的指针。对于数组 int arr[5] = {1, 2, 3, 4, 5};
,数组名 arr
在传递时等价于指针 &arr[0]
,即数组第一个元素的地址。
1.2.2 arr[i]
表示什么?
在 printArray
函数中,当你使用 arr[i]
时,C++ 编译器会将其解释为指针算术和解引用操作。具体来说:
arr
是一个指向int
类型的指针,指向数组的第一个元素。i
是一个整数索引,用于访问数组中的元素。
表达式 arr[i]
可以分解为以下步骤:
- 指针算术:
arr + i
计算指针arr
加上偏移量i
。由于arr
是一个int*
类型的指针,加上i
后得到一个新的指针,指向数组中的第i
个元素。指针算术会自动根据指针类型调整偏移量。例如,如果arr
是int*
类型,每次增加1
意味着增加sizeof(int)
字节。 - 解引用:
*(arr + i)
解引用该指针,访问指针所指向的内存位置的值,即数组中的第i
个元素。
所以,arr[i]
实际上等价于 *(arr + i)
。
2. 二维数组
#include <iostream>
// 函数声明,形参为一个二维数组和一个整数
void print2DArray(int arr[][3], int rows) {
// 遍历二维数组并打印每个元素
for (int i = 0; i < rows; ++i) {
for (int j = 0; j < 3; ++j) {
std::cout << arr[i][j] << " ";
}
}
}
int main() {
// 定义一个二维数组并初始化
int arr[2][3] = {{1, 2, 3}, {4, 5, 6}};
// 调用函数,传递二维数组和行数
print2DArray(arr, 2);
return 0;
}
2.1数组退化
二维数组退化:二维数组 arr[2][3]
在传递给函数时,会退化为指向包含3个整数的数组的指针,即 int (*arr)[3]
。这意味着 arr
是一个指向 int[3]
类型数组的指针。
访问元素:
arr[i]
:指向第i
行的数组,类型是int[3]
。arr[i][j]
:指向第i
行第j
列的元素。根据指针算术和解引用操作,这个表达式可以分解为*(*(arr + i) + j)
。arr + i
:指向第i
行的指针,类型是int (*)[3]
。*(arr + i)
:指向第i
行的数组,类型是int[3]
。*(arr + i) + j
:指向第i
行第j
列的元素的指针,类型是int*
。*(*(arr + i) + j)
:解引用这个指针,得到第i
行第j
列的元素。
2.2 示例解释
假设 arr
是 int arr[2][3] = {{1, 2, 3}, {4, 5, 6}}
并且 rows
是 2
:
- 外层循环
i
:- 当
i = 0
时:- 内层循环
j
:- 当
j = 0
时:arr[0][0]
等价于*(*(arr + 0) + 0)
,值是1
。
- 当
j = 1
时:arr[0][1]
等价于*(*(arr + 0) + 1)
,值是2
。
- 当
j = 2
时:arr[0][2]
等价于*(*(arr + 0) + 2)
,值是3
。
- 当
- 内层循环
- 当
i = 1
时:- 内层循环
j
:- 当
j = 0
时:arr[1][0]
等价于*(*(arr + 1) + 0)
,值是4
。
- 当
j = 1
时:arr[1][1]
等价于*(*(arr + 1) + 1)
,值是5
。
- 当
j = 2
时:arr[1][2]
等价于*(*(arr + 1) + 2)
,值是6
。
- 当
- 内层循环
- 当
最终,print2DArray
函数会打印二维数组的所有元素:
1 2 3
4 5 6
2.3 总结
- 在函数参数中,二维数组
int arr[][3]
退化为int (*arr)[3]
,即指向包含3个整数的数组的指针。 arr[i]
指向第i
行的数组,类型是int[3]
。arr[i][j]
等价于*(*(arr + i) + j)
,访问第i
行第j
列的元素。- 通过双重循环,可以遍历和打印二维数组中的所有元素。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 【杭电多校比赛记录】2025“钉耙编程”中国大学生算法设计春季联赛(1)
2019-07-16 Numpy数据类型转化astype,dtype
2019-07-16 支持向量机(三)非线性支持向量机与核函数