问:
第11行,*(*p2+i)为啥里面还要加*号呢?*p2+i不就是里面的值了吗?问了一下人是双层指针,实在是不懂#include <stdio.h>
int main(void)
{
int temp[5] = {5,22,35,4656,515};
int (*p2)[5] = &temp;//int (*p2)[5] = temp,指针指向第一个数组元素的地址,实际上指向整型变量的指针
int i;
for(i=0;i<5;i++)
{
printf("%d\n",*(*p2+i));//多取了一次地址,所以p2也要加*号
}
return 0;
}
答:
如果想真正全面理解为什么会如此,首先要理解一下数组名:
数组名的本意是表示一组数据的集合,它和普通变量一样,都用来指代一块内存。如果你打印sizeof(temp),结果为20(在32位机器下为20,也就是4个int类型的宽度)。
但在使用过程中,数组名有时候会转换为指向数据集合的指针(地址),而不是表示数据集合本身。
这是因为:C语言标准规定,当数组名作为数组定义的标识符(也就是定义或声明数组时)、sizeof 或 & 的操作数时,它才表示整个数组本身,在其他的表达式中,数组名会被转换为指向第 0 个元素的指针(地址)。请一定注意,除了上面的三种情况(数组定义的标识符、sizeof 或 & 的操作数)外,数组名将会转换为指向第0个元素的指针。
也就是说,数组名在这三种外,数组名temp本身的类型为int*;而在这三种情况内,你使用取地址符&或检查长度sizeof时代表的是数组本身。所以你打印sizeof(temp),结果为20;你&temp的类型为int[5]*。即
int* p1=temp; //p1指向的类型为int
int (*p2)[5]=&temp; //p2指向的类型为int[5]
所以,p2实际上是一个二维数组指针。
下面,我来进一步解释为什么p2是一个二维数组指针。二维数组在概念上是二维的,有行和列,但在内存中实际上所有的数组元素都是连续排列的,它们之间没有“缝隙”。比如二维数组
int a[3][4] = { {0, 1, 2, 3}, {4, 5, 6, 7}, {8, 9, 10, 11} };
在内存中,a 的分布是一维线性的,整个数组占用一块连续的内存。就是从0开始,连续地址存放到11。也就是
0 1 2 3 4 5 6 7 8 9 10 11
C语言允许把一个二维数组分解成多个一维数组来处理。对于数组 a,它可以分解成三个一维数组,即 a[0](可以理解为一个新的名字)、a[1](可以理解为一个新的名字)、a[2](可以理解为一个新的名字)。每一个一维数组又包含了 4 个元素,例如 a[0] 包含 a[0][0]、a[0][1]、a[0][2]、a[0][3]。
为了更好的理解指针和二维数组的关系,我们先来定义一个指向 a 的指针变量 p:
到此,我们可以理解为什么p2是一个而维数组指针了。
|