美文网首页计算机C语言相关思考
关于二维数组及数组指针和指针数组的深度思考(涉及指针)

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

作者: cseeing | 来源:发表于2018-05-25 12:51 被阅读0次

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

    • 数组指针
      定义 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.

    相关文章

      网友评论

        本文标题:关于二维数组及数组指针和指针数组的深度思考(涉及指针)

        本文链接:https://www.haomeiwen.com/subject/wywyjftx.html