指针数组名加1和数组的数组的数组名加1的区别?
本帖最后由 小蒙 于 2017-11-4 19:11 编辑char *p = {"hh", "xx"};
printf("p is %p\n", p);
printf("p + 1 is %p\n", p + 1);
printf("p = %p\n", p);
printf("addr of \"hh\" is %p\n", "hh");
printf("p + 1 = %p\n", p + 1);
结果
p is 0x7fff5a02daa0
p + 1 is 0x7fff5a02daa8
p = 0x105bd2f10
addr of "hh" is 0x105bd2f10
p + 1 = 0x105bd2f11
注意,p和p+1的值差了8,p和p的地址不同。
以下为个人浅见,p是一个指针数组,按照之前学数组的知识, p作为数组名,值是指针数组首元素的地址,但是不能认为这个首元素是"wm",因为,指针数组,指针数组,所以里面的元素都是指针,所以首元素也应该是个指针,所以,首元素的地址就是指针的地址,即指针自身的地址,和指针指向的元素("hh")无关,对比p和p+1发现,确实地址增加了8,正好是一个指针的大小;事实也证明p的值,即指针的地址不同于"hh"的地址;再看p,p是二维数组的第一个元素,相当于一维数组数组名,所以它的值是一维数组数组第一个字符的地址,对比p和p+1发现,二者相差1,正好是一个char的大小,指针数组分析完毕,下面来看数组的数组。
char b[] = {"hh", "xx"};
printf("%p\n", b);
printf("%p\n", b + 1);
printf("%p\n", &b);
printf("%p\n", &b + 1);
结果
0x7fff50f19a70
0x7fff50f19a7a
0x7fff50f19a70
0x7fff50f19a84
经过上面的分析,我已经不敢说b到底代表什么了,抛开定义方式导致的静态内存和动态内存的区别,指针数组和数组的数组代表的值字面上是一样的,二者写法上是“互换的”,所以用指针数组的那一套分析方法放在这个数组上发现跟结果对不上,b会代表指向"hh"指针的地址么???事实证明b和b+1相差10,正好是单个元素的宽度,所以,b应该代表的是"hh"作为一个一维数组的首地址(即一维数组的首地址)(看到结果确实好理解。。。),所以加一地址才会增加宽度个大小,至于&b和&b+1差了20是因为&b代表二维数组的首地址,&b加一,地址会增加2个宽度为10的大小。 难道不同的定义方法,看到数组名的时候在思考上也是不同的么? 何必纠结这么多呢,我觉得关于数组名和指针的关系,只要记住两点就行了:
1 数组名是一个常量,所以不能做自增自减等类似操作
2 数组形式始终只是一种语法糖,都可以用指针进行直接转换,除了上面说的1 {:10_257:}没有明白你想表达的意思... 我自己的问题,把指针数组char *p当成了二维的了,结帖。 橙C 发表于 2017-11-4 18:28
没有明白你想表达的意思...
刚才混乱了,把指针数组看成二维的了。 本帖最后由 小蒙 于 2017-11-12 22:06 编辑
橙C 发表于 2017-11-4 18:28
没有明白你想表达的意思...
刚才看了一下帖子,发现之前的理解是错的。我之前真正的疑问是下面定义方式产生的结果不同。char *p = {"hh", "xx"};
打印p和p+1的值,相差8,即指针的宽度
char b[] = {"hh", "xx"};
打印b和b+1的值,相差10,即一个元素的宽度。
刚才想明白了,p是指针数组,花括号中的字符串实际上是存在于静态存储区,指针数组p只是存储了两个指针,分别指向静态存储区的两个字符串而已,这样的定义使p存储的实际是两个指针,并且分别指向"hh"和"xx"对应的静态存储区,感觉p还是应该看成一个二维数组,数组名p代表首元素的地址,两个元素都是指针所以,所以p代表的是指针的自身的地址,所以p+1是下一个指针的地址,所以相差8;b是按数组形式定义的二维数组,所以{"hh", "xx"}实际上有两个,一个存在静态存储区,另一个是b复制的,b指向的实际上是复制的那个,所以首元素是字符串,而不是指针,所以b代表的是"hh"作为二维数组一个元素的首地址,所以b+1和b相差元素个宽度,所以相差10。
页:
[1]