C语言二维数组及数组指针、指针数组

先复习一下数组的定义

例如:定义一个33列的整型数组

  • int array[3][3]

上述代码声明了一个 3 行 3 列的二维数组,其 数组名 是 array,其下标变量的类型为整形。

如下图所示:该数组的下标变量一共有 3x3 个。

先来看一下这道题,第一个*ptr和*(ptr+1), 第二个**ptr和**(ptr+1)的值分别是什么,看不懂,然后就看下面的介绍。这道题是C  primer  plus 第10章的课后习题,对于我们理解二维数组和指针帮助大,需要理解,提出疑问,  int *ptr;ptr = torf[0];和 int  (*ptr)[2]; ptr = torf;?

    int *ptr;
    int torf[2][2] = {{1 } , 3 ,4};
    ptr = torf[0];
    printf("*ptr = %d, *(ptr + 1) = %d\n", *ptr, *(ptr + 1));



    
    int  (*ptr)[2];
    int  torf[2][2] = {12, 14 , 16};
    ptr = torf;
    printf("ptr =%d, **(ptr + 1) = %d\n", **ptr, **(ptr+1));

这是4行2列的数组,需要好好理解

int a[4][2] = {{2,4},{6,8},{1,3},{5,7}};
    printf("a = %p, a+1 = %p\n",a, a+1);
    printf("a[0] = %p, a[0]+1 = %p\n", a[0], a[0]+1);
    printf("*a = %p, *a+1 = %p\n", *a, *a+1);
    printf("a[0][0] =%d\n", a[0][0]);
    printf("*a[0] = %d\n", *a[0]);
    printf("**a = %d\n", **a);
    printf("a[2][1] = %d\n",a[2][1]);
    printf("*(*(a+2)+1)=%d\n", *(*(a+2)+1));

1)如何识别二维数组。 假设数组为a[2][5]
二维数组肯定是有两个维度,从左到右看:
[2]是第一个维度,表示a这个数组里有两个元素。
[5]是第二个维度,需要进入内部观察。第一个维度的每个元素的内部有5个int型的元素。


2)“数组名代表数组首元素的地址”这句话既适用于一维数组,也适用于二维数组。
1、对于一维数组int a[5]而言:数组名a就表示首元素a[0]的地址,及数组名a等价于&a[0];

2、对于二维数b[2][5]组而言:数组名b就表示首元素b[0]的地址,及数组名b等价于&b[0];

3、对于二维数b[2][5]组而言:b[0]是第一维数组的首元素,同时b[0]也是第二维中的数组的数组名。
所以数组名b[0]就表示首元素b[0][0]的地址,及数组名b[0]等价于&b[0][0];

4、对于二维数b[2][5]组而言:b[1]虽然不是第一维数组的首元素,但也是第二维中的数组的数组名。
所以数组名b[1]就表示首元素b[1][0]的地址,及数组名b[1]等价于&b[1][0];

3)访问二维数组的两种方式
1、普通指针指向二维数组的第一维
由于二维数组的,第一维度都是普通的一维数组。如b[2][5]中b[0]和b[1]就都是一维的int型数组。
而b[0]和b[1]都是数组名,分别代表b[0][0]和b[1][0]的地址,所以指针指向二维数组的第一维很容易:
int* p1 = b[0];
int* p2 = b[1];
那么此时,用法和一维数组一样:
printf("b[0][0] = %d.\n", *p1);//*p1对应的是b[0][0]的值
printf("b[0][1] = %d.\n", *(p1+1));//*(p1+1)对应的是b[0][1]的值

printf("b[1][0] = %d.\n", *p2);//*p2对应的是b[1][0]的值   
printf("b[1][1] = %d.\n", *(p2+1));//*(p2+1)对应的是b[1][1]的值

2、数组指针访问二维数组
    对于二维数b[2][5]组而言:数组名b就表示首元素b[0]的地址,及数组名b等价于&b[0];
而b[0]不仅仅是二维数b[2][5]的首元素,同时也是一个数组的数组名。

    所以,&b[0]就相当于对数组名再次区地址。那么我们想想&b[0]是个什么类型?

    假设有一个一维数组int a[5],数组a表示a[0]地址,那么继续对数组名取地址——&a,&a表示什么呢?
我们都知道&a此时数组上和a相同,但意义上不同,&a表示整个数组的地址(而不再是首元素的地址)。

    这时数组指针就可以登场了,数组指针就是用来指向整个数组的地址。数组指针的类型为 int (*)[]。
所以&b[0]以及&a都是和这种类型相匹配的,或者说他们就是这种类型。

    回到一开始说的,二维数b[2][5]的数组名b等价于&b[0],而&b[0]是对数组名再次取地址,及&b[0]和int (*)[]类型匹配。
所以有:
    int (*p)[5] = b;//这里要注意的是数字5,因为二维数组b[2][5],第二维是5,所以这里数组指针也是5;数字不同则不匹配。
    我们知道,取址和解引用这两个过程是反向过程,&b[0]是对地址(数组名)再次取地址,那么可以想象,要用数组指针p访问到数据,
必须经过两次解引用
    由于数组名b就表示首元素b[0]的地址,而数组名b和数组指针p是同一种类型,所以第一层解引用就是对b[0]的地址解引用,
及是对第一维数组的地址解引用。如&b[0]解引用之后就得到b[0],那么*p就对应b[0],而b[0]仍然是个数组名,再次进行解引用得到b[0][0],
那么*(*p)就对应b[0][0]。最后得出结论:b[i][j]等同于 *(*(p+i)+j)

小结:

1、 a[i][j]等同于 *(*(p+i)+j)

2、 p不解引用,对p加减是在第一维里偏移地址

3、 p解一次引用,对*p加减是在第二维里偏移地址

4、 p解二次引用,才能访问到值,如* (*p)

指针数组,一个很好理解的demo,可以自己在虚拟机跑一下

首先char *a [ ]是一个指针数组,理解下面例子有两点基础:

  1. 数组的名就是数组的地址
  2. printf()接收到地址会将地址上存的连续的字符串打印出来
#include <stdio.h>

int main()
{
    char *a [ ] = {"China","French","America","German"};

    printf("数组所在地址:");
    printf("%p \n",a); 

    printf("数组中第一个元素所在,即数组地址:");
    printf("%p \n",&a[0]); 

    printf("数组中元素(指针所指地址):");
    printf("%p %p %p %p\n",a[0],a[1],a[2],a[3]);
    
    printf("数组中第一个元素(指针)所指地址:");
    printf("%p \n",a[0]); 
    
    printf("数组中第一个元素(指针)所指地址中存放的元素(只是字符串的第一个字符):");
    printf("%c \n",*a[0]); 
    
    printf("数组地址取值(第一个元素,即第一个指针地址)对应的字符串:");
    printf("%s \n",*a); 

    printf("数组地址移位取值(第二个元素,即第二个指针地址)对应的字符串:");
    printf("%s \n",*(a+1)); 

    printf("数组第一个元素(指针)所指位置上对应的字符串:");
    printf("%s \n",a[0]); 

    printf("数组第一个元素(指针)所指位置上对应的字符串中第一个字符:");
    printf("%c \n",a[0][0]); 

    printf("数组第二个元素(指针)所指位置上对应的字符串中第二个字符:");
    printf("%c \n",a[1][1]); 

    return 0;

}

来源:(34条消息) 数组指针与二维数组详解_裸奔の猪的博客-CSDN博客_二维数组和指针

           (36条消息) 关于char *a [ ]的讲解例子_char *a[]_Peneus的博客-CSDN博客

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值