数组指针和指针数组的区别 一、指针数组和数组指针的内存布局 初学者总是分不出指针数组与数组指针的区别。其实很好理解:指针数组:首先它是一个数组,数组的素都是指针,数组占多少个字节由数组本身的大小决定,每一个素都是一个指针,在32 位系统下任何类型的指针永远是占4 个字节。它是“储存指针的数组”的简称。数组指针:首先它是一个指针,它指向一个数组。在32 位系统下任何类型的指针永远是占4 个字节,至于它指向的数组占多少字节,不知道,具体要看数组大小。它是“指向数组的指针”的简称。下面到底哪个是数组指针,哪个是指针数组呢:A)int *p1[10];B)int (*p2)[10];每次上课问这个问题,总有弄不清楚的。这里需要明白一个符号之间的优先级问题。“[]”的优先级比“*”要高。p1 先与“[]”结合,构成一个数组的定义,数组名为p1,int *修饰的是数组的内容,即数组的每个素。那现在我们清楚,这是一个数组,其包含10 个指向int 类型数据的指针,即指针数组。至于p2 就更好理解了,在这里“()”的优先级比“[]”高,“*”号和p2 构成一个指针的定义,指针变量名为p2,int 修饰的是数组的内容,即数组的每个素。数组在这里并没有名字,是个匿名数组。那现在我们清楚p2 是一个指针,它指向一个包含10 个int 类型数据的数组,即数组指针。我们可以借助下面的图加深理解:
二、int (*)[10] p2—–也许应该这么定义数组指针 这里有个有意思的话题值得探讨一下:平时我们定义指针不都是在数据类型后面加上指针变量名么?这个指针p2 的定义怎么不是按照这个语法来定义的呢?也许我们应该这样来定义p2: int (*)[10] p2;int (*)[10]是指针类型,p2 是指针变量。这样看起来的确不错,不过就是样子有些别扭。其实数组指针的原型确实就是这样子的,只不过为了方便与好看把指针变量p2 前移了而已。你私下完全可以这么理解这点。虽然编译器不这么想。^_^ 三、再论a 和&a 之间的区别 既然这样,那问题就来了。前面我们讲过a 和&a 之间的区别,现在再来看看下面的代码:int main(){ char a[5]={‘A’,’B’,’C’,’D’}; char (*p3)[5] = &a; char (*p4)[5] = a; return 0;}上面对p3 和p4 的使用,哪个正确呢?p3+1 的值会是什么?p4+1 的值又会是什么?毫无疑问,p3 和p4 都是数组指针,指向的是整个数组。&a 是整个数组的首地址,a是数组首素的首地址,其值相同但意义不同。在C 语言里,赋值符号“=”号两边的数据类型必须是相同的,如果不同需要显示或隐式的类型转换。p3 这个定义的“=”号两边的数据类型完全一致,而p4 这个定义的“=”号两边的数据类型就不一致了。左边的类型是指向整个数组的指针,右边的数据类型是指向单个字符的指针。在Visual C++6.0 上给出如下警告: warning C4047: ‘initializing’ : ‘char (*)[5]’ differs in levels of indirection from ‘char *’。还好,这里虽然给出了警告,但由于&a 和a 的值一样,而变量作为右值时编译器只是取变量的值,所以运行并没有什么问题。不过我仍然警告你别这么用。既然现在清楚了p3 和p4 都是指向整个数组的,那p3+1 和p4+1 的值就很好理解了。但是如果修改一下代码,把数组大小改小点,会有什么问题?p3+1 和p4+1 的值又是多少呢?int main(){ char a[5]={‘A’,’B’,’C’,’D’}; char (*p3)[3] = &a; char (*p4)[3] = a; return 0;}甚至还可以把代码再修改,把数组大小改大点:int main(){ char a[5]={‘A’,’B’,’C’,’D’}; char (*p3)[10] = &a; char (*p4)[10] = a; return 0;}这个时候又会有什么样的问题?p3+1 和p4+1 的值又是多少?上述几个问题,希望读者能仔细考虑考虑,并且上机测试看看结果。 测试结果: (1).char (*p2)[5]=a;必须使用强制转换,如:char (*p2)[5]=(char (*)[5])a; (2).把数组大小改变,都会编译不通过,提示: error C2440: ‘initializing’ : cannot convert from ‘char (*)[5]’ to ‘char (*)[3]’ error C2440: ‘initializing’ : cannot convert from ‘char (*)[5]’ to ‘char (*)[10]’ (3).把以上程序测试代码如下: int main() { char a[5]={‘a’,’b’,’c’,’d’}; char (*p1)[5]= &a; char (*p2)[5]=(char (*)[5])a; printf(“a=%d ”,a); printf(“a=%c ”,a[0]); printf(“p1=%c ”,p1); printf(“p2=%c ”,p2); printf(“p1+1=%c ”,(p1+1)); printf(“p2+1=%c ”,(p2+1)); return 0; } 输出: a= a=a p1=a p2=a p1+1=? p2+1=? Press any key to continue 结论: 根据指针类型及所指对象,表示指针大小,每次加1,表示增加指针类型大小的字节.—-后面还会有解释说明.
2024最新激活全家桶教程,稳定运行到2099年,请移步至置顶文章:https://sigusoft.com/99576.html
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。 文章由激活谷谷主-小谷整理,转载请注明出处:https://sigusoft.com/45641.html