精华篇:数组指针

精华篇:数组指针一.数组指针1.定义:数组指针,指的是数组名的指针,即数组首元素地址的指针。即是指向数组的指针。例:int(*p)[10];p即为指向数组的指针,又称数组指针。2.特性:…

大家好,又见面了,我是你们的朋友全栈君。

一.数组指针

1.定义:

         数组指针,指的是数组名的指针,即数组首元素地址的指针。即是指向数组的指针。

         例:int (*p)[10]; p即为指向数组的指针,又称数组指针。

2.特性:

         数组指针用于指向一个数组,其本质为指针;

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

         数组指针也叫行指针

3.二维数组指针:

         int (*p)[5]

         p是一个指针变量,它指向包含5个int元素的一维数组,此时p的增量以它所指向的一维数组长度为单位;

        *(p+i)是一维数组a[i][0]的地址;

        *(p+2)+3表示a[2][3]地址(第一行为0行,第一列为0列),*(*(p+2)+3)表示a[2][3]的值。

4.用法用例:

(1).一维数组指针

#include<stdio.h>

int main(){
    
    int a[12] = {10,20,30,40,50,60,70,80,90,100,110,120};
    
    int (*p_array)[12] = &a;                //指向一维数组的数组指针
                                            //a 是一个int [12] 类型的数组, 取它的地址初始化p_array. 值得注意的是此时数组名array不再表示数组首元素的地址.
    int *p = a;                                //指向一维数组元素的指针
    
    printf(“%d\n”,p[0]);
    printf(“%d\n”,*p_array[0]);
    
    printf(“%d\n”,a[1]);
    printf(“%d\n”,*(p + 1));
    printf(“%d\n”,p[1]);
    printf(“%d\n”,(*p_array)[1]);    
    printf(“%d\n”,p_array[0][1]);    
                                            /*
                                            (1) 二维数组实际上就是元素为一维数组的数组
                                            (2) 二维数组名可以看做指向其第一个元素(一维数组)的指针
                                            (3) 现在我们将array看作一个只有一个元素的二维数组,
                                            并且该二维数组的元素是 int [5] 类型的,
                                            显然arrayptr指向了该二维数组的第一个元素(相当于二维数组名)
                                            所以我们可以通过下标运算符 arrayptr[0]
                                            获得二维数组的第一个元素(实际上就是array数组),
                                            然后再次利用下标运算符 arrayptr[0][0] 获取array数组的第一个元素.
                                            注意到无法使用 arrayptr[1]是错误的, 因为该二维数组只有一个元素.
                                            */
    
    return 0;
}

 

<span role="heading" aria-level="2">精华篇:数组指针

(2).二维数组指针

#include<stdio.h>

int main(){
    
    int a[3][4] = {10,20,30,40,50,60,70,80,90,100,110,120};
    
    printf(“%d\n”,a[0][1]);
    int (*p_array)[4] = a;                //指向一维数组的数组指针 ,此时p_array指向的是a数组的第一个元素(元素是数组)
                                        //等价与 int* p_array = &a[0]
    printf(“%d\n”,p_array[0][1]);       //同理,p_array【0】指向的是 a[0][]这个数组
    printf(“%d\n”,(*p_array)[1]);       //[]的优先级要比*要高,所以要加括号
                                        /*在一维数组中,
                                        数组名表示的是数组第一个元素的地址,
                                        那么二维数组呢?
                                        a 表示的是元素 a[0][0] 的地址吗?
                                        不是!我们说过,二维数组就是一维数组,
                                        二维数组 a[3][4] 就是有三个元素 a[0]、a[1]、a[2] 的一维数组,
                                        所以数组 a 的第一个元素不是 a[0][0],
                                        而是 a[0],所以数组名 a 表示的不是元素 a[0][0] 的地址,而
                                        是 a[0] 的地址*/
                                        
    printf(“%d\n”,*((*p_array) + 1));   //a[0][]的地址的下一个地址自然就是a[0][1]了
    
                                        //那么我们要如何访问a[2][1]这个元素呢
                                        
    printf(“%d\n”,a[2][1]);
    printf(“%d\n”,p_array[2][1]);
    printf(“%d\n”,(*p_array)[9]);                        //从p_array[0][0]开始,第十个元素就是a[2][1] (2 * 4 + 2 = 10)
    printf(“%d\n”,*((*p_array) + 9));                     //等价上面的代码
    printf(“%d\n”,p_array[0][9]);                        //等价上面的代码
    printf(“%d\n”,(*(p_array + 2))[1]);                   //注意数组指针是行指针,他得改变是针对一个一维数组改变的
    printf(“%d\n”,*(*(p_array + 2) + 1));                 //*(p_array + 2)他表示的是a[2][0]的地址,它的下一个地址自然就是a[2][1]了
                                                        //从上面的代码中我们可以看得出,访问数组元素,要么用下标法,即[],要么用指针法
    printf(“%d\n”,((p_array + 1)[1])[1]);               //(p_array + 1)[1])等价与p_array[2]
    printf(“%d\n”,((p_array + 1) + 1)[0][1]);             //p_array + 2等价于a[2],第一个[0]位于内层数组标号,
                                                        //还是按行指针加法规则,意思是相对于a[2][]偏移0个行指针,即还是a[2][],
                                                        //第二个下标[1]    是行指针内的元素
    
    printf(“%d\n”,(*((p_array + 1) + 1))[1]);             //因为是行指针,所以最内层的p_array每加一按一个数组的长度加(即到了第三个数组,下标为2)
                                                        //等价与上面代码
    
    
    return 0;
}

 

<span role="heading" aria-level="2">精华篇:数组指针

关于指针数组,我们只需掌握一个规则:

(1).访问数组元素,要么用下标法 [],要么用指针*.

(2).一维数组用一个下标,二维数组用两个下标

(3).指针指向数组元素时,指针表示数组元素的地址,用 * 对这个地址运算就求出了这个地址(即数组元素)的值

(4).指针数组指向二维数组时,因为它本身是个指向数组的指针,所以它的内层每加一,就相当于下一个数组

          如 array[m][n],int (*p)[n] = a

          本来p是指向 a[0][] 这个数组的,*p 本身就相当于a[0],p[0]

          p + 1 就指向了二维数组的分数组的下一个数组,实际是 m + 1,即 p + 1 >> m + 1,*(p + 1) >> a[m + 1]

          而,如果想让数组指针指向当前分数组的下一个元素,我们需要让它的 n + 1

          即 (* p) + 1 == &a[0][1], *((*p) + 1) == a[0][1]

(5),我们回想一下,在

         int a[3][4]   ,int(*p)[4] = a

         中 a 代表 a分数组的第一个数组的地址,即 a >> &a[0],  而a[0] >> &a[0][0]

         所以  a 间接 与 &a[0][0],实际效果是相同的,但我们不能说它们等价,因为等价的意思是符号表达的意义是相同的

         但这里它们意义是不相同的,只能说它们的值相等

         而 我们说过,指针数组是指向数组首地址的指针,所以

         p >> a ,而 a >> &a[0], 所以, p >> &a[0]

         所以很自然的我们能够得出

         p[m] >> a[m], *p >> a[0] >> *a

         *(p + m) >> a[m] >> *(a + m),

         *(*(p + m) + n) >> (*(p + m))[n] >> a[m][n] >> p[m][n]

         <1>.第一步中的m,对于a来说,是a总数组的第m个分数组,

         但对于p来说,它是指指向当前数组的下m个数组的首地址,虽然它们等价,

         但是这实际上是因为沾了(数组内存是连续)的光,如果二维数组的分数组地址是不连续的,

         那么这个结论自然也就不成立,

         所以,即使它们等价,我们还是要理解的它们实际意义的区别

         <2>.从第一步以后的推论,我们可以发现一个重要的事情:

         p >> a

         即,是p的地方,我们就可以换成a

         是a的地方,我们就能换成p

         这是因为

         a 代表 &a[0],  p指向a中第一个分数组的首地址,即&a[0],

         注意是数组的首地址,不是数组第一个元素的首地址,即使它们的地址是相同的

         综上所述,虽然 p >> a,而且是完全等价,但是,我们还是要理解它们的区别

         因为对于

         a[m][n], int(*p)[n] = a

         这一切都是 a >> &a[0] 造成的影响

         若写成 p = &a[m], 你还懂么?

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/155794.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 机器学习之Python — Sklearn库简介

    机器学习之Python — Sklearn库简介文章目录机器学习之Python—Sklearn库简介1Sklearn简介2Sklearn安装3Sklearndatasets4Sklearn通用学习模式5Sklearn数据预处理–标准化6Sklearn交叉验证7总结参考资料机器学习之Python—Sklearn库简介1Sklearn简介Scikit-learn(sklearn)是机器学习中常用的第三方模块,对常…

    2022年10月11日
    0
  • json数组转对象集合_解析json字符串转成对象集合

    json数组转对象集合_解析json字符串转成对象集合json数组转对象//使用alibabafastjson@Testpublicvoidtest3(){Stringstr=”[{‘aa’:’123′,’bb’:’456′},{‘aa’:789,’bb’:’112′}]”;List<Test1>tests=JSONArray.parseArray(str,Test1.class);for(Test1test:tests){System.

    2022年9月11日
    0
  • windows定时任务schtasks命令详细解

    SCHTASKS/Create[/Ssystem[/Uusername[/P[password]]]][/RUusername[/RPpassword]]/SCschedule[/MOmodifier][/Dday][/Mmonths][/Iidletime]/TNtaskname/TRtaskrun[/STstarttim…

    2022年4月4日
    293
  • Spring框架介绍及使用

    Spring框架介绍及使用Spring框架—控制反转(IOC)1Spring框架概述1.1什么是Spring1.2Spring的优点1.3Spring的体系结构2入门案例:(IoC)2.1导入jar包2.2目标类2.3配置文件2.4测试3入门案例:DI3.1目标类3.2dao3.3service3.4配置文件3.5测试4依赖注入…

    2022年6月18日
    17
  • NSGA2算法中拥挤度计算代码[通俗易懂]

    NSGA2算法中拥挤度计算代码[通俗易懂]思想:    要对拥挤距离进行计算,则需要根据每个目标函数对种群中的所有个体按升序进行排序。第一个和最后一个个体的拥挤距离设为无穷大,第i个个体的拥挤距离则设为第i+1和第i个体的所有目标函数值之差的和。具体方法如下面伪代码:defcrowding_distance_assignment(I)nLen=len(I)#I中的个体数量…

    2022年5月19日
    43
  • 252个核心词根——词缀(前缀-后缀)总结大全【最全-一文看懂!!!】[通俗易懂]

    252个核心词根——词缀(前缀-后缀)总结大全【最全-一文看懂!!!】[通俗易懂]目录:一:什么是词根词缀?1.图解2.词缀3.词根4.补充(1)通假【清辅音浊化,浊辅音清化】(2)辅音主表意(3)元音主发音(4)五组通假关系(5)英语单词记忆八大原理5.词根词缀大全https://www.youdict.com/root/root.php二:学词根词缀有必要吗?记单词步骤三:252个常用英语词根记忆法背单词252词…

    2022年7月24日
    29

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号