指针和多维数组

数组名是特殊的指针

数组是一个特殊的指针,多维数组也是更为复杂的数组,它们的关系是什么样的呢?

我们通过一个简单的例子来比较形象的了解指针和多维数组:

int a[2][3];

这是一个2*3的二维数组,首先我们清楚数组名就是指向数组首元素的常量指针(它不可以指向其他部分,可以对指向的元素进行任意修改);其次C语言中所谓的多维数组,即是数组的数组,2*3的二维数组,本质上为2个有包含3个int的数据的数组。所以现在我们就可以解释a的含义:

a == &a[0]

那么对于a[0]也有

a[0] == &a[0][0]

这时候我们就也可以得到另一个特殊的结论:a == a[0],字面上看起来很难理解,这是因为对于一个指针它指向一个元素有两个要素:第一为这个元素的首地址,第二为这个元素的类型(这也是我们在使用指针所必要要求的必须指向与它类型相同的元素),所以a是一个指向包含3个int元素的数组指针,a[0]是指向int元素的指针。在比较两者时即比较首地址,无疑都是这个二维数组的首地址。

 #include <stdio.h>
 ​
 int main (void) {
  int a[2][3];
  if (a == &a[0])
      printf("yes!\n");
  if (a[0] == &a[0][0])
      printf("yes!\n");
  if (a == a[0])
      printf("yes!");
  return 0;
 }

但是两者仍然有差别,这种差别是在类型上的,也可谓是根本上的,通过指针的增加运算我们可以看到他们的不同:

 #include <stdio.h>
 ​
 int main (void) {
  int a[2][3];
  int *p = a;
  int *pp = a[0];
  if (a == a[0])
      printf("yes!\n");
  if (a+1 == a[0]+1)
      printf("yes!\n");
  else {
      printf("no!\n");
      printf("a = %p\n", a);
      printf("p = %p\n", a+1);
      printf("pp = %p", a[0]+1);
  }
 ​
  return 0;
 }
 ​
 /**
  yes!
  no!
  a = 0061FF00
  p = 0061FF0C
  pp = 0061FF04
  因为a为int (*)[3]类型,所以a+1移动了3*4=12个储存单元,末位为C;a[0]为int *类型,所以a[0]+1移动了4个单元,这时候它们不再相等。
 **/

通过指针指向多维数组

因为C语言要求指针在指向一个元素时类型必须同元素一致,所以我们想使用指针指向数组时不能简单使用指向int类型的指针。例如对于二维数组a[2][3]

 int main (void) {
  int a[2][3];
  int (*p)[3] = a;
  // int (*p1)[2] = a;这样是不对的,因为C语言二维数组的声明意义为2个长度为3的整型数组而不是相反的
  return 0;
 }

我们使用int (*p)[3]而不是int *p[3],因为前者表示指向含有三个整数数组的指针,后者为含有3个整型指针的数组。

我们也能通过解引用这样的指针对于数组内容进行访问。

我们通过讨论数组名这个特殊的指针来了解指向多维数组的指针,a == &a[0],所以*a = a[0]为第一个指向长度为3数组的指针;a[0] == &a[0][0],所以*a[0] = a[0][0]这样我们就访问了二维数组的第一个元素,所以**a == a[0][0],我们可以通过多次解引用对多维数组元素进行访问,对于二维数组a[1][2]等价于*(*(a+1)+2)

但是通过这样对一个数组进行访问往往会造成费解,而且有时候会出现不安全的现象:

 #include <stdio.h>
 ​
 int main (void) {
  int a = 20;
  const int b = 30;
  int *p1 = &a;
  const int *p2 = &b;// 指向常量的指针指向常量
  const int **pp2;// 指向 指向常量的指针 的指针
  // p1 = p2;// 使用普通指针指向指向常量的指针这样的行为并不可行!
  p2 = p1;// 把普通指针赋给指向常量的指针时时没有问题的
 ​
  // 下面是一种值得警惕的情况
  printf("b = %d\n", b);
  pp2 = &p1;// 这样的行为也是不安全的,称为为嵌套指针类型赋值,这样会使得const修饰符失效
  *pp2 = &b;
  *p1 = 10;
  printf("b = %d", b);
  return 0;
 }
 ​
 /**
  b = 30
  b = 10
 **/

我们惊奇发现b竟然也被修改了!因为我们让pp2指向指针p1。然后因为pp2是指向常量的指针,所以*pp2也可以指向常量b,但是其实*pp2是p1!这时候我们就能通过p1修改常量b的值,这样行为是很危险的!所以不要使用嵌套指针。

C语言中指针和多维数组的更多相关文章

  1. c语言中指针和多维数组的理解

    1.复习指针和数组之间的特殊关系:不带方括号的数组名是一个指针,指向该数组的第一个元素. 2.多维数组: ][];//声明一个二维数组作为举例 a.理解方式1:可以将数组看成行和列构成,即理解成2行4 ...

  2. 以杨辉三角为例,从内存角度简单分析C语言中的动态二维数组

    学C语言,一定绕不过指针这一大难关,而指针最让人头疼的就是各种指向关系,一阶的指针还比较容易掌握,但一旦阶数一高,就很容易理不清楚其中的指向关系,现在我将通过杨辉三角为例,我会用四种方法从内存的角度简 ...

  3. C语言中如何将二维数组作为函数的参数传递

    今天写程序的时候要用到二维数组作参数传给一个函数,我发现将二维数组作参数进行传递还不是想象得那么简单里,但是最后我也解决了遇到的问题,所以这篇文章主要介绍如何处理二维数组当作参数传递的情况,希望大家不 ...

  4. 论C语言中二级指针和二维数组之间的区别

    刚开始学习C语言的时候,觉得一个数组可以定义一个一级指针去访问,想当然的就觉得可以定义一个二级指针去访问二维数组.很显然这是错误的. 我们来看看C语言的数组在内存中的存储方式. 实际上C语言中的数组, ...

  5. C语言中指针和数组

    C语言数组与指针的那些事儿 在C语言中,要说到哪一部分最难搞,首当其冲就是指针,指针永远是个让人又爱又恨的东西,用好了可以事半功倍,用不好,就会有改不完的bug和通不完的宵.但是程序员一般都有一种迷之 ...

  6. 图解c/c++多级指针与“多维”数组

    声明:本文为原创博文,如有转载,请注明出处.若本文有编辑错误.概念错误或者逻辑错误,请予以指正,谢谢. 指针与数组是C/C++编程中非常重要的元素,同时也是较难以理解的.其中,多级指针与“多维”数组更 ...

  7. 图解C/C++多级指针与多维数组

    声明:本文转自 chenyang_yao ,欢迎阅读原文. 指针与数组是C/C++编程中非常重要的元素,同时也是较难以理解的.其中,多级指针与“多维”数组更是让很多人云里雾里,其实,只要掌握一定的方法 ...

  8. 编程基础-c语言中指针、sizeof用法总结

    1.指针 学习 C 语言的指针既简单又有趣.通过指针,可以简化一些 C 编程任务的执行,还有一些任务,如动态内存分配,没有指针是无法执行的.所以,想要成为一名优秀的 C 程序员,学习指针是很有必要的. ...

  9. <转载>c++中new一个二维数组

    原文连接 在c++中定义一个二维数组时有多种方式,下面是几种定义方式的说明:其中dataType 表示数据类型,如int  byte  long... 1.dataType (*num)[n] = n ...

随机推荐

  1. Educational Codeforces Round 89 (Rated for Div. 2)D. Two Divisors 线性筛质因子

    题目链接:D:Two Divisors 题意: 给你n个数,对于每一个数vi,你需要找出来它的两个因子d1,d2.这两个因子要保证gcd(d1+d2,vi)==1.输出的时候输出两行,第一行输出每一个 ...

  2. Atcoder Beginner Contest 168 D - .. (Double Dots) (BFS)

    题意:有\(n\)个房间,在这些房间中两两连\(m\)次条边,问除了第一个房间,其他房间走到第一个房间的最短路径,输出这个房间所连的上一个房间,如果走不到,输出\(no\). 题解:刚开始我写了一个d ...

  3. Codeforces Round #515 (Div. 3) C. Books Queries (模拟)

    题意:有一个一维的书架,\(L\)表示在最左端放一本书,\(R\)表示在最右端放一本书,\(?\)表示从左数或从右数,最少数多少次才能得到要找的书. 题解:我们开一个稍微大一点的数组,从它的中间开始模 ...

  4. .net webapi 中使用session是出错 HttpContext.Current.Session==null

    最近在写.net webapi时发现 HttpContext.Current.Session==null  ,导致报错,后来查资料发现webapi中使用session时首先需要开启session功能, ...

  5. 踏上Revit二次开发之路 3 自己的工具按钮

    3 自己的工具按钮 上次的例子只能在"附加模块"→"外部工具"下运行,用作个人作品是没问题,如果打算搞个公司产品的话,估计BOSS是不会满意的.这次我来做一个直 ...

  6. 四、Jmeter 集合点(实际场景应用)

    一.jmeter集合点的作用域及作用范围 先明确一些概念:1)定时器是在每个sampler(采样器)之前执行的,而不是之后: 是的,你没有看错,不管这个定时器的位置放在sampler之后,还是之下,它 ...

  7. C++ part2

    为什么析构函数必须是虚函数?为什么C++默认的析构函数不是虚函数? references: nowcoder 将可能会被继承的父类的析构函数设置为虚函数,可以保证当我们new一个子类,然后使用基类指针 ...

  8. SpringBoot 全局视角看springboot

    从单体架构到微服务 单体架构 任何一个网站在发布初期几乎都不可能立马就拥有庞大的用户流量和海量数据,都是在不停 的试错过程中一步一步演变其自身架构,满足其自身业务.比如现在能够抗住双十一这么大 流量的 ...

  9. SSH Keys vs GPG Keys

    SSH Keys vs GPG Keys SSH Keys SSH keys allow you to establish a secure connection between your compu ...

  10. ADN vs CDN All In One

    ADN vs CDN All In One Netlify & JAMstack https://app.netlify.com/teams/xgqfrms/sites ADN Applica ...