小白问题,关于指针数组

#include
void main()
{
char*p[4]={"apple","pear","peach","orange"};
for(int i=1;i<4;i++)
printf("No.%d = %s" , i , p[i]);
}
其中数组p的每一个元素不都是一个指针么?
为什么在输出时输出p[i]而不是*p[i]???
(输出p[i]不是应该输出各指针所指向的字符串的地址么?)

关键是 printf("No.%d = %s" , i , p[i]); 这句。

例子:
char a[20] = "hello world";
printf("%s", a);

上面的a就是数组的地址,也就是说printf中‘%s’,输出字符串时需要的参数是字符串的地址。
所以
printf("No.%d = %s" , i , p[i]); 输出的是p[i]指向的字符串是正常的。

想要输出各指针所指向的字符串的地址,应该这样写:
printf("%0x", p[i]); //%p或者 %0x都行,指针通常用16进制表示

而*p[i]是没有意义的。p[0]是第一个字符串的地址,p[0][0]是第一个字符串的第一个字符,你的例子里就是'a';

相当与p[0]也是‘apple’数组的地址,*p[0]的内容应该也是'a',相当与*(p[0] + 0)也就是p[0][0]; 但这样写可读性差也不合逻辑,因为p[i]是一个数组的指针。

指针数组:array of pointers,即用于存储指针的数组,也就是数组元素都是指针
int* a[4] 指针数组

表示:数组a中的元素都为int型指针

元素表示:*a[i] (a[i])是一样的,因为[]优先级高于

应该用*p[i]而不是p[i]

p[i]输入的是地址,*p[i]才是内容

就这个程序来看,printf语句输出的字符串用p[i]时可以正确输出,这是怎么回事,求解答?

p[i]取的是char*类型字符串,该字符串是存储在字符串常量区,读取时会根据char*的地址读取字符串,直到遇到'\0'结束;
而*p[i]取的是char类型的字符,根据返回类型可以看出差异。

p[i] :是字符串,是地址;
*p[i]:是内容,是p[i]字符串中的第一个字符值

输出时跟printf这个函数有关 为%s时 会找到指向字面值常量字符串的首地址,然后依次去读取后面的内容,知道碰到结束符 '\0'为止,
即p相当于一个char** 型的二级指针。