数组是连续内存的别名,指针是存储地址的变量;数组名是常量地址不可修改,指针是变量可改变指向;数组名在表达式中常退化为指向首元素的指针,但sizeof运算中数组返回总字节数,指针返回地址大小;函数参数中“数组”实际是指针,需额外传长度。

很多C++初学者在学习数组和指针时,常常被它们相似的语法和行为搞混。其实,数组和指针在本质上完全不同,理解它们的关键在于搞清楚“是什么”和“怎么用”。
数组是连续内存的别名,指针是地址的变量
当你定义一个数组,比如 int arr[5];,编译器会分配一块连续的内存空间,用来存放5个整数。这个 arr 本质上是一个指向这块内存起始位置的常量地址,不能被修改。也就是说,arr 不是一个变量,它代表的是内存块本身的“名字”。
而指针,比如 int* p;,是一个真正的变量,它存储的是某个内存地址。你可以改变这个地址,让它指向不同的位置。指针本身也有自己的内存空间(比如在64位系统上占8字节)。
数组名在大多数情况下会“退化”成指针
当你在表达式中使用数组名时,比如传参或赋值,它通常会自动转换成指向第一个元素的指针。这就是为什么你可以写 p = arr;。但这只是“用法上的便利”,不代表数组就是指针。
立即学习“C++免费学习笔记(深入)”;
关键区别体现在:
- sizeof(arr) 返回整个数组的字节数(如5 * 4 = 20),而 sizeof(p) 只返回指针本身的大小(通常是8)。
- &arr 和 arr 数值相同,但类型不同:&arr 是指向整个数组的指针(int(*)[5]),arr 是指向第一个元素的指针(int*)。
- 你不能对数组名赋值,比如 arr = p; 是非法的,因为 arr 是常量地址。
函数参数中的“数组”其实是伪装的指针
当你写函数 void func(int arr[]) 或 void func(int arr[10]),编译器会自动将其视为 void func(int* arr)。这意味着你传进去的数组会退化成指针,函数内部无法通过 sizeof(arr) 得到数组长度。
所以,通常需要额外传一个长度参数:
void printArray(int* arr, int size) {for (int i = 0; i std::cout }
}
从内存布局理解两者关系
假设你有:
int data[3] = {10, 20, 30};int* ptr = data;
内存中:
- data 占据12字节,存放三个整数。
- ptr 占据8字节(64位),存放的是 data 的地址(即 &data[0])。
虽然 data[i] 和 ptr[i] 都能访问相同元素,但前者是基于固定内存块的偏移,后者是基于指针变量的解引用。
基本上就这些。理解数组是“一块内存的名字”,指针是“保存地址的变量”,就能避免大多数混淆。多写代码,用 sizeof 和取地址符验证,印象会更深刻。










