c语言指针的有关问题

#include<studio.h>
int main(){
int *p,i,j;
int a[][]={1,2,3,4,5,6,7,8,9};
p=*a;
for(i=0;i<3;i++)
for(j=0;j<3;j++)
printf("%d",(&p)[i][j]);
return 0;
}
上面这段代码为何无法遍历输出 a数组的全部元素?这种方式访问a数组有什么错误?

a是3*3的二维数组
    int a[3][3] = {1, 2, 3, 4, 5, 6, 7, 8, 9};
也就等同是3个长度为3的一维数组
所以p应该是指向长度为3的一维数组的指针
    int (*p)[3];
这样p才能用p[i][j]两个下标.
p[i]返回的是一维数组的地址, 其中i每增加1,指针就向后移动3个int类型字节长度地址
[j]是在p[i]指针移动后的基础上再向后移动j个int类型字节长度

你用int *p就是指向一个int类型字节长度地址
这样只能用p[i]一个下标.
p[i]返回的是数值,不是地址,就不能用第二个下标[j]
#include <stdio.h>
#include <stdlib.h>
int main()
{
    int (*p)[3], i, j;
    int a[3][3] = {1, 2, 3, 4, 5, 6, 7, 8, 9};
    p = *a;
    for (i = 0; i < 3; i++)
        for (j = 0; j < 3; j++)
            printf("%d", p[i][j]);
    return 0;
}

如有帮助,请点击我的回答下方的【采纳该答案】按钮帮忙采纳下,谢谢!

img

理解和讨论之前请先学会如何观察!

//char (*(*x[3])())[5];//x是什么类型的变量?
//
//分析C语言声明,关键是搞清楚这个变量是个什么东西(函数、指针、数组),
//是函数那么剩下的就是他的参数和返回值,
//是指针那剩下部分是说明他指向什么,
//是数组剩下的部分就是说明数组的成员是什么类型。
//解析C语言声明规则:
//从左侧第一个标识符开始,按照优先级进行结合。*表示是..的指针,const表示只读的,volatile表示可变的,[]表示是数组,()表示是函数。
//
//x和[3]结合说明是一个大小为3的数组,该数组的每个元素为一类指针,该类指针指向一类函数,该类函数无参数,返回一类指针,该类指针指向一个大小为5的char型数组
#include <stdio.h>
#include <typeinfo.h>
char num[5];
char (*x00())[5] {
    return &num;
}
int main() {
    char (*(*x[3])())[5];//是个数组,大小为3
    char (*(*x0  )())[5];//数组的元素,是个函数指针
    char (*( x00 )())[5];//函数原型,参数为空,返回值为指针
    char (*  x000   )[5];//返回值

    x0 = x00;
    x[0] = x0;
    x[1] = x0;
    x[2] = x0;
    printf("typeid(x).name() is %s\n",typeid(x).name());
    return 0;
}
//typeid(x).name() is char (* (__cdecl**)(void))[5]