关于二维数组及数组指针和指针数组的深度思考(涉及指针)

转载请注明
在纠结指针数组和数组指针时无意发现的小细节。
总算搞清数组指针和指针数组。

  • 数组指针
    定义 int (*p)[n];
    ()优先级高,首先说明p是一个指针,指向一个整型的一维数组,这个一维数组的长度是n,也可以说是p的步长。也就是说执行p+1时,p要跨过n个整型数据的长度。

二维数组可以定义

int a[3][4]={1,2,3,4,5,6,7,8,9,10,11,12} ;
p=a;//是将二维数组a[3][4]的首地址给了p,即a[0]也可视为a[0][0]
p+=2;//不是到a[0][2]而是到a[2][0]

int (*p)[4]={1,2,3,4,5,6,7,8,9,10,11,12};//这样定义和int ((*p)+4)这样定义都是报错的

  • 指针数组
    定义 int *p[n];
    []优先级高,先与p结合成为一个数组,再由int*说明这是一个整型指针数组,它有n个指针类型的数组元素。这里执行p+1时,则p指向下一个数组元素,这样赋值是错误的:p=a;因为p是个不可知的表示,只存在p[0]、p[1]、p[2]…p[n-1],而且它们分别是指针变量可以用来存放变量地址。但可以这样 *p=a; 这里*p表示指针数组第一个元素的值,a的首地址的值。
    eg:

int *p[3];//此数组每一个元素都是一个指针
int a[3][4];
p++; //表示p数组指向下一个数组元素。
for(i=0;i<3;i++)
p[i]=a[i];
这里int *p[3] 表示一个一维数组内存放着三个指针变量,分别是p[0]、p[1]、p[2]

重点(指针和指针变量不同):
系统为每一个内存单元分配一个地址值,C/C++把这个地址值称为“指针”。如有int i=5;,存放变量i的内存单元的编号(地址)&i被称为指针。
“指针变量”则是存放前述“地址值”的变量,也可以表述为,“指针变量”是存放变量所占内存空间“首地址”的变量(因为一个变量通常要占用连续的多个字节空间)。比如在int i=5;后有一句int p=&i;,就把i的指针&i赋给了int 型指针变量p,也就是说p中存入着&i。所以说指针变量是存放指针的变量。
有一个事实值得注意,那就是有不少资料和教科书并没有如上区分,而是认为“指针是指针变量的简称”,如对int p=&i;的解释是:声明一个int 型指针p,并用变量i的地址初始化;而严格说应该是声明一个int *型指针变量p才对。

要表示数组中i行j列一个元素时:(p[i]+j)、((p+i)+j)、((p+i))[j]、p[i][j]
优先级:()>[]>*

在做下面这个题的时候产生了思考(题目要求打出指定号码的人的成绩(人的号码:0~3))

#include <stdio.h>
int main(void)
{
    double score[][4] ={{60.0,70.0,80.5,90.5},{56.0,89.0,67.0,88.0},{34.2,78.5,54.0,34.0},{20.0,45.0,89.0,90.0}};
    double *search(double(*pointer)[4],int n);
    double *p;
    double(*pointer)[4];
    int i,n,m;

    printf("please enter a number of student:");
    scanf("%d",&m);

    printf("The score of No.%d are :\n",m);

    p=search(score,m);
    for(i=0;i<4;i++)
        printf("%5.2f\t",*(p+i));
    printf("\n\n\n");
 } 
 double *search(double(*pointer)[4],int n)
 {
    double *pt;
    int i;
    pt =*(pointer + n);
    for(i=0;i<4;i++)
        printf("%5.2f\t",*(*(pointer + n)+i));
    printf("\n");
    return pt;
 }

在函数中加入printf观察两个指针的输出结果有什么不同

输入 1
输出
The score of No.1 are :
56.00 89.00 67.00 88.00
56.00 89.00 67.00 88.00

而后将函数中的
for(i=0;i<4;i++)
printf("%5.2f\t",*(*(pointer + n)+i));

改为
for(i=0;i<4;i++)
printf("%5.2f\t",*((pointer + n)+i));

输入 1
输出
The score of No.1 are :
0.00 0.00 0.00 0.00
56.00 89.00 67.00 88.00

再将其改为
for(i=0;i<4;i++)
printf("%5.2f\t",*((pt + n)+i));

输入 1
输出
The score of No.1 are :
89.00 67.00 88.00 34.20
56.00 89.00 67.00 88.00

发现按道理应该打出a[1][4]但没有a[1][4],应该打出0.00,但是打出34.20。
分析得出
数组在内存条的储存方式为按顺序存储,所以a[1][3]完了应该是a[2][0],所以打出34.2.

猜你喜欢

转载自blog.csdn.net/qq_40077114/article/details/80444888