C语言指针指向二位数组的疑问?
int (*p)[3] = data;
printf("%d\n",p);
printf("%d\n",*p);
printf("%d %d %d %d %d %d\n",(*p)[0],(*p)[1],(*p)[2],(*p)[3],(*p)[4],(*p)[5]);
printf("%d %d %d %d %d %d\n",*(p[0]),*(p[1]),*(p[2]),*(p[3]),*(p[4]),*(p[5]));
printf("%d %d %d %d %d %d\n",data[0],data[1],data[2],data[3],data[4],data[5]);
输出如下:
-2147456552
-2147456552
111 222 333 444 555 666
111 444 -2147456552 222 825307407 875836448
-2147456552 -2147456540 -2147456528 -2147456516 -2147456504 -2147456492
我的理解这样定义后,p是表示数组第一行{111 222 333}的地址,*p是表示数组第一行第一列元素111的地址,这样两者相同,都是-2147456552
请问:
(1)*(p[2])为何也是-2147456552?它怎么会指向第一行(第一列)?
(2)根据“int (*p)[3] ”,定义的(*p)长度为3,(*p)[3],(*p)[4],(*p)[5]应当都是非法的吧?怎么会有值?还是依次的
(3)(*p)[n]与*(p[n])到底什么区别?值不同应怎么解释?
谢谢高人解答! 展开
一条一条来
(1)*(p[2])为何也是-2147456552?它怎么会指向第一行(第一列)?
*(p[2])就是 data[2][0]; 对于data数组来说,它已经越界了,它读取的是data数组最后一个元素的下一个地址内的数据,在这个例子里,这个地址恰好就是p的空间,而p保存的就是data的地址,所以一样输出data的地址,注意这是恰好,给p分配的内存恰好就紧挨着data,不同的编译器,不同环境下,不一定是这样
(2)根据“int (*p)[3] ”,定义的(*p)长度为3,(*p)[3],(*p)[4],(*p)[5]应当都是非法的吧?怎么会有值?还是依次的
data的空间是连续的 (*p)[3] 就是data[0][3] ,也就是 data[1][0],对于data[0]来说,它越界了,对于整个data来说,没越界
(3)(*p)[n]与*(p[n])到底什么区别?值不同应怎么解释?
这样解读:(*p)[n] , 有括号,所以p与*结合,那么p就是个指针,然后发现有[n], 那么就说明这个指针指向一个一维数组,所以p是个指针,它指向一个长度是n的一维数组
*(p[n]); 这也就是*p[n] ; p先和[]结合,说明p是个数组,然后再与*结合,说明这个数组的元素是* , 也就是说,p是个数组,它有n个元素,每个元素都是指针
1、*(p[2])其实是一个没初始化的元素,你的int (*p)[3]定义的p是一个指向数组的指针,这个数组中有三个int元素,所以*(p[0])就是第一个数组的第一个元素(也就是data[0][0]),*(p[1])是第二个数组的第一个元素(也就是data[1][0]),实际上你要访问data[i][j],可以通过printf("%d\n",*(*(p+i)+j));
2、C里对指针越界没有明确的检查,发生越界时程序可以正常也可能崩溃。
3、前者定义的是一个指针,后者定义的是一个数组,具体请参照网页链接