当我阅读有关C语言的书籍时,这个两级指针给我带来了很多困扰。
char s[5][5];
char *s[5];
char (*s)[5];
那么他们之间有什么区别?
答案 0 :(得分:16)
在C中,最好说出声明。然后,它变得直观。为此,您遵循左右惯例。这是怎么回事:
char *s[5];
你如何发言它?为此,您从变量名s
的右侧开始,然后转到左侧。所以你先说“s是a / an”,在右边,你会看到一个[]
,你说“s是一个数组......”。然后你走到左边看*
,然后说“s是一个指针数组。”这就是它。它是一个5指针的数组。您可以在此数组中存储不同的指针。
现在换另一个:
char (*s)[5];
你以同样的方式开始。 “s是a / an”,然后查看()
。 ()
内的任何内容都比{}}更接近s
。因此,*
与s
的关系比[]
更紧密。所以你现在说,“s是一个指针......”现在你走出括号并看到[]
。所以你继续说,“s是一个指向数组的指针”。而这正是它的本质。它是一个指针,指向数组的第一个元素。
现在遵循相同的逻辑并尝试猜测以下内容:
int (*callme)(int a, int b)
int (*callme[10])(int a, int b)
提示,最后一个可用于为函数创建查找表。
编辑:
正如评论中所提到的,开头还有一个char
。我从来没有能够找到一种简单的方式来说明这一点,但从上下文中可以清楚地看出来。例如,在第一个示例中,char
定义了数组的类型,而在第二个示例中,它定义了指针。在我发布的练习中,int
定义了函数返回值的类型。通常使用这些定义,将只有一个具有未定义类型的项。这就是我如何弄清楚类型的去向。
答案 1 :(得分:8)
答案 2 :(得分:3)
虽然涵盖了declaration,但也可能会指出用法的差异:
char s[5][5];
-
s
指向已分配内存的区域(如果是全局堆,则为本地堆栈),s[0][0]
s[4][4]
(或s[0]
.. s[24]
)安全地写出最多25个字符值,sizeof(s)
== 25。 char *s[5];
-
s
指向已分配内存的区域(如果是全局堆,则为本地堆栈),s[0]
.. s[4]
,sizeof(s)
== 40(*)。 char (*s)[5];
-
s
不指向任何已分配的内存 - 此时它只是一个未初始化的指针,&s
,sizeof(s)
== 8(*)。(*)注意:假设采用64位架构。
答案 3 :(得分:3)
1.char s [5] [5];
这里是具有5行和5列的二维数组。在这5行和5列中,您将保存字符类型的元素。
2.char * s [5]; s是一个包含5个元素的一维数组,每个元素都是字符指针的类型。
3.char(* s)[5]; s是这里的指针而不是数组。 S指向一个字符数组。例如。
char arr[5][5];
char(*s)[5];
s = arr;
s [0] [0]将与arr [0] [0]
的数组相同