基础C++教学⮱⮱009【C中通过指针引用多维数组】2019-12-22

←↑→↓↖↙↗↘↕⏤
unicode=Geometric Shapes
▶ 仅仅个别字不同的时候的对比标识

◉ 着重强调

  ◆ 1、
  ◆ 2、
  ◆ 3、

  
  
  
  
  
  
  
  

Miscellaneous Symbols


Dingbats
✍ 重点记忆,个人总结的点,或者知识。
✎✎

    int a[3][3]= { {1,2,3},{6,5,4},{7,8,9}};

    printf("a[0]=%x\n",a[0]);
    printf("a[1]=%x\n",a[1]);
    printf("a[2]=%x\n",a[2]);

    printf("%c",'\n');

    printf("a=%x\n",a);
    printf("a+0=%x\n",a+0);
    printf("a+1=%x\n",a+1);
    printf("a+2=%x\n",a+2);
a[0]=61fedc
a[1]=61fee8
a[2]=61fef4

a=61fedc
a+0=61fedc
a+1=61fee8
a+2=61fef4

  用visual studio运行查看内存:

  数组首地址为0x00d5fd5c,因为一个int占4字节, 3个int就是12字节,等于0x0c。
  0x00d5fd5c+0x0c=0x00d5fd68正好等于a[1]a[2]的值计算方式相同。
  结论:a[0]、a[1]、a[2],各为二维数组第一维的起始地址。

  需要注意的点:
  a[0]=61fedc 和 a+0=61fedc,值相同,但是含义相同吗?
  如何验证呢??

    printf("a[0]+1=%x\n",a[0]+1);
    printf("a+0+1=%x\n",a+0+1);
a[0]+1=61fee0
a+0+1=61fee8

  根据地址输出,可以看到同样都执行了操作+1,但是得出的结果却不同,这说明了虽然a[0]和a+1输出值相同,但是却代表不同含义。
  下图简要的标识了两者之间的不同关系。

  回顾一维数组的取值方式:

  明确:

  1、二维数组,要取值,要使用2对[ ][ ]符号。

  2、从一维数组来说,方括号[ ]和星号*的作用相同。一维数组要取值,要么表达式使用[ ],要么使用*

  3、从二维数组来说,常规使用2对[ ][ ],上所述[ ]*作用相同,所以理论上二维数组取值可以使4种方式:

  下面使用程序进行验证四种方式:

     printf("a[0][0]=%x\n",a[0][0]);
     printf("*(a[0]+0)=%x\n",*(a[0]+0));
     printf("*(a+0)[0]=%x\n",*(a+0)[0]);
     printf("*(*(a+0)+0)=%x\n",*(*(a+0)+0));
a[0][0]=1
*(a[0]+0)=1
*(a+0)[0]=1
*(*(a+0)+0)=1

  到目前为止,情况一切都很美好,但是,还不够全面。
  只验证了第一维的第一个元素,后面的情况如何?
  1、[ ][ ]式

    printf("a[0][0]=%x\n",a[0][0]);
    printf("a[0][1]=%x\n",a[0][1]);
    printf("a[0][2]=%x\n",a[0][2]);
a[0][0]=1
a[0][1]=2
a[0][2]=3

  2、[ ]*式 ( [ ] 在括号内,先进行运算 )

    printf("*(a[0]+0)=%x\n",*(a[0]+0));
    printf("*(a[0]+1)=%x\n",*(a[0]+1));
    printf("*(a[0]+2)=%x\n",*(a[0]+2));
*(a[0]+0)=1
*(a[0]+1)=2
*(a[0]+2)=3

  3、* *式

    printf("*(*(a+0)+0)=%x\n",*(*(a+0)+0));
    printf("*(*(a+0)+1)=%x\n",*(*(a+0)+1));
    printf("*(*(a+0)+2)=%x\n",*(*(a+0)+2));
*(*(a+0)+0)=1
*(*(a+0)+1)=2
*(*(a+0)+2)=3

  4、* [ ]式

    printf("*(a+0)[0]=%x\n",*(a+0)[0]);
    printf("*(a+0)[1]=%x\n",*(a+0)[1]);
    printf("*(a+0)[2]=%x\n",*(a+0)[2]);
*(a+0)[0]=1
*(a+0)[1]=6
*(a+0)[2]=7

  嗯。?????

  这???出现了超出预期的结果。对比数组的定义如下:

int a[3][3]= { {1,2,3},{6,5,4},{7,8,9}};

  出现的结果,变成了3个维度的各第一个元素。这和预计的初衷不同。
  反思问题在哪?对的,应该是 运算符优先级
  根据c语言语法,*(a+0)[0]这个表达式中,各运算的优先级为:

  这意味着:

  原本初衷是访问第一维的3个元素,针对这个需求,如何修改代码?
  根据上表逻辑和原本需求,要访问第一维的3个元素,需要取值表达式的[ ]符号作为这个表达式的最后一个执行的部分,或者说,需要把基址“固定住”。
  或者说,既然是因为优先级导致的问题,那么就从改变优先级入手,添加一对括号:

  这意味着:下面再看上表的计算过程:

  使用程序输出验证:

    printf("(*(a+0))[0]=%x\n",(*(a+0))[0]);
    printf("(*(a+0))[1]=%x\n",(*(a+0))[1]);
    printf("(*(a+0))[2]=%x\n",(*(a+0))[2]);
(*(a+0))[0]=1
(*(a+0))[1]=2
(*(a+0))[2]=3
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 指针是C语言中广泛使用的一种数据类型。 运用指针编程是C语言最主要的风格之一。利用指针变量可以表示各种数据结构; ...
    朱森阅读 8,814评论 3 44
  • 谨记 每个人心中都有一片海,自己不扬帆,没人帮您启航,久了就是一片死海。人生,就是一场自己与自己的较量:让积极打败...
    长风留言阅读 12,435评论 6 10
  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 13,150评论 1 51
  • 数组在程序设计中,为了处理方便, 把具有相同类型的若干变量按有序的形式组织起来。这些按序排列的同类数据元素的集合称...
    朱森阅读 9,562评论 2 13
  • 第十章 指针 1. 地址指针的基本概念: 在计算机中,所有的数据都是存放在存储器中的。一般把存储器中的一个字节称为...
    坚持到底v2阅读 4,733评论 2 3