指针下标要求数组或指针类型什么意思*a[]和指向指针的指针**a有什么关系?

数组指针与指针数组的区别在于:数组指针p是一个指针,而指针数组p是一个存放N个指针变量的数组。
用于指向一个二维数组。
重点:()优先级高([]、()的优先级是一样的,但它们的方向是从左至右的,所以先运行括号里的*p),首先说明p是一个指针,指向一个整型的一维数组,n是指一位数组的长度,在二维数组中列数就是每一个一位数组的长度,行数是一位数组的个数。也可以说是p的步长。也就是说执行p+1时,p要跨过n个整型数据的长度(n*sizeof(int))。
如要将二维数组赋给一指针,应这样赋值:
int (*p)[4]; //该语句是定义一个数组指针,指向含4个元素的一维数组。
同时用来指向二维数组时,其引用和用数组名引用都是一样的即a<=>p。比如要表示数组中i行j列一个元素a[i][j]:

重点:[]优先级高,先与p结合成为一个数组,再由int*说明这是一个整型指针数组,它有n个指针类型的数组元素:即它就是一个存放了n个指针的数组。
这样赋值也是错误的:p=a;因为p是个右值,p的值只存在p[0]、p[1]、p[2]…p[n-1],而且它们分别是指针变量可以用来存放变量地址。但可以这样*p=a; 这里*p表示指针数组第一个元素的值,a的首地址的值。
如要将二维数组赋给一指针数组:
这里int *p[3] 表示一个一维数组内存放着三个指针变量,分别是p[0]、p[1]、p[2],默认情况下这三个指针变量都指向NULL的,所以要分别赋值。

}

如果一个数组中的所有元素保存的都是,那么我们就称它为指针数组。其一般形式为:

它是一个数组,数组的元素都是指针,数组占多少个字节由数组本身的大小决定,每个元素都是一个指针。

 例如:char *arr[]={“Sunday”,“Monday”},存储了两个指针,第一个指针指向了字符串"Sunday",第二个指针指向了字符串"Monday",而sizeof(arr)=8,因为在32位平台,指针类型大小占4个字节。指针数组最重要的用途是对多个字符串进行处理操作,因为字符指针比二维数组更快更有效。

//定义一个存放指向整型变量的指针的数组arr //通过接引用打印出三个一维数组的元素

以上对arr解引用的方式有很多,它们都是等价的,我们来举个例子:

//以下均为不同方式的解引用操作

结果如下所示: 

不同解引用操作的结果为:

从以上例子可看出解引用有多种方式,它们的等价形式如下:

补充(1)指针数组还可以和字符串数组相结合使用,请看下面的例子:

需要注意的是,字符数组 str 中存放的是字符串的首地址,不是字符串本身,字符串本身位于其他的内存区域,和字符数组是分开的。

也只有当指针数组中每个元素的类型都是char *时,才能像上面那样给指针数组赋值,其他类型不行。

 为了便于理解,可以将上面的字符串数组改成下面的形式,它们都是等价的。

补充(2):二维数组与指针数组的区别



注:因为数组指针对于一维数组的使用比较尴尬,对于一维数组,建议使用指针数组比较方便,这里只涉及到关于二维数组与数组指针的知识!!!

首先引入二维数组的定义:二维数组在概念上是二维的,有行有列,但在内存中所有的元素都是连续排列的,以下面的二维数组为例:

从概念上理解,a的分布就像一个矩阵:

 从内存上理解,整个数组占用一块连续的内存:

 C语言中的二维数组是按行排列的,也就是先存放 a[0] 行,再存放 a[1] 行,最后存放 a[2] 行;每行中的 4 个元素也是依次存放。数组 a 为 int 类型,每个元素占用 4 个字节,整个数组共占用 4×(3×4) = 48 个字节。

C语言允许把一个二维数组分解成多个一维数组来处理。对于数组 a,它可以分解成三个一维数组,即 a[0]、a[1]、a[2]。每一个一维数组又包含了 4 个元素,例如 arr[0] 包含 a[0][0]、a[0][1]、a[0][2]、a[0][3]。

假设数组a中第0个元素的地址为1000,那么每个一维数组的首地址如下图所示:

 为了更好的理解和二维数组的关系,我们先来定义一个指向 a 的指针变量 p:

括号中的*表明 p 是一个指针,它指向一个数组,数组的类型为int [4],这正是 a 所包含的每个一维数组的类型。

[]的优先级高于*()是必须要加的,如果赤裸裸地写作int *p[4],那么应该理解为int *(p[4]),p 就成了一个指针数组,而不是二维数组指针。

对指针进行加法(减法)运算时,它前进(后退)的步长与它指向的数据类型有关,p 指向的数据类型是int [4],那么p+1就前进 4×4 = 16 个字节,p-1就后退 16 个字节,这正好是数组 a 所包含的每个一维数组的长度。也就是说,p+1会使得指针指向二维数组的下一行,p-1会使得指针指向数组的上一行。数组名 a 在表达式中也会被转换为和 p 等价的指针!

 下面我们就来探索一下如何使用指针 p 来访问二维数组中的每个元素。按照上面的定义:


1) p指向数组 a 的开头,也即第 0 行;p+1前进一行,指向第 1 行。


2) *(p+1)表示取地址上的数据,也就是整个第 1 行数据。注意是一行数据,是多个数据,不是第 1 行中的第 0 个元素,下面的运行结果有力地证明了这一点:

*(p+1)单独使用时表示的是第 1 行数据,放在表达式中会被转换为第 1 行数据的首地址,也就是第 1 行第 0 个元素的地址,因为使用整行数据没有实际的含义,编译器遇到这种情况都会转换为指向该行第 0 个元素的指针;就像一维数组的名字,在定义时或者和 sizeof、& 一起使用时才表示整个数组,出现在表达式中就会被转换为指向数组第 0 个元素的指针

4) *(*(p+1)+1)表示第 1 行第 1 个元素的值。很明显,增加一个 * 表示取地址上的数据

根据上面的结论,可以很容易推出以下的等价关系:

 【实例】使用指针遍历二维数组。


指针数组和二维数组指针在定义时非常相似,只是括号的位置不同:

指针数组和二维数组指针有着本质上的区别:指针数组是一个数组,只是每个元素保存的都是指针,以上面的 p1 为例,在32位环境下它占用 4×5 = 20 个字节的内存。二维数组指针是一个指针,它指向一个二维数组,以上面的 p2 为例,它占用 4 个字节的内存。

}

首先声明,我是一个菜鸟。一下文章中出现技术误导情况盖不负责

        既然指针数组明白了,那个数组指针就只能是 int (*a)[10]了,我们可以以为 (*a)是一个数组名,那么这个数组有10个int型的数据,只是这个数组名是一个指针,所以这叫数组指针。数组指针也叫行指针,如

美丽是平凡的,平凡得让你感觉不到她的存在;美丽是平淡的,平淡得只剩下温馨的回忆;美丽又是平静的,平静得只有你费尽心思才能激起她的涟漪。

        这样两者的区别就恍然大悟了,数组指针只是一个指针变量,好像是专门用来指向二维数组的,它据有内存中一个指针的存储空间。指针数组是多个指针变量,以数组情势存在内存当中,据有多个指针的存储空间。

文章结束给大家分享下程序员的一些笑话语录: 看新闻说中国输入法全球第一!领先了又如何?西方文字根本不需要输入法。一点可比性都没有。

}

我要回帖

更多关于 下标要求数组或指针类型什么意思 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信