C语言中的声明解析规则——数组,指针与函数
2013-09-18 20:05
232 查看
摘要:C语言的申明存在的最大问题是:你无法以一种人们所习惯的自然方式和从左向右阅读一个声明,在引入voliatile和const关键字以后,情况更加糟糕了。由于这些关键字只能出现在声明中,是的声明形式和使用形式完全对上号的例子越来越少了。而C语言中比较绕人的指针数组和数组指针的问题,int *ap[]和int (*ap)[]谁是指针数组,谁又是数组指针?这里面声明的解析规则是什么样的?本文主要为你解答这些疑惑。
我们来看看下面的语句,和它们对应的编译结果:
int (*ap)[2]={1,2};
/*stringcat.c:6:3: warning: initialization makes pointer from integer without a cast [enabled by default]
14 stringcat.c:6:3: warning: (near initialization for ‘ap’) [enabled by default]
15 stringcat.c:6:3: warning: excess elements in scalar initializer [enabled by default]
16 stringcat.c:6:3: warning: (near initialization for ‘ap’) [enabled by default]
17 */
int (*ap)[2]=int(*)[2]a;
18 /*stringcat.c:7:16: error: expected expression before ‘int’
20 */
int a[2]={1,2}
int (*ap)[2]=(int(*)[2])a;//注意,此处申明的不是一个数组而是一个指针
注意,这个才是正确的,但是你能看出上面声明的是一个指针而不是一个数组吗?
指针和数组之所以复杂,主要原因就再次:他们的声明和使用形似并不相同。来看看C语言的声明是如何形成的?
声明器就是标识符以及与它组合在一起的任何指针,函数括号和数组下标等。
构造声明要遵循以下条件:
函数返回值不能是一个函数:foo()()
函数返回值不能是一个数组:foo()[]
数组里面不能有函数:foo[]()
但可以这样:
函数可以返回函数指针:int (*fun())()
函数可以返回指向数组的指针:int(*foo())[]
数组里面允许函数指针:int(*foo[])()
数组里面允许其他数组:int foo[][]
这里面,比较qipa的有三种类别,数组,函数,指针。
一般定义:int a;
数组:int a[5];
函数:int fun();
指针:int *p;
尤其是数组和函数,我们可以认为他们的变量处在类型修饰符(不过把运算符也算作类型的话)的中间。
我们接下来看看C语言的优先级规则
A:声明从名字开始读取,然后按照优先级次序依次开始读取
B:优先级从高到低的次序是
B1:声明中被括号扩起来的部分
B2:后缀操作符() []
B3:前缀操作符*
C:const和volatile关键字后面紧跟类型说明符的话(int,long),那么它作用与类型说明符号;反之,它作用与左边挨着的*
如此,可以分析char * const * (*next)()
具体的分析过程按照上述即可,结果是:next是一个指针,它指向一个函数,这个函数返回另一个指针,该指针是指向类型为char的常量指针。
然后再来分析以往我们提出的例子就简单多了:int (*ap)[2],ap是一个指针,这个指针指向长度为2的int类型的数组;int *ap[2],ap是一个长度为2的数组,数组中的每个元素是指向int类型的指针。这样就明晰了数组指针和指针数组的区别。
我们来看看下面的语句,和它们对应的编译结果:
int (*ap)[2]={1,2};
/*stringcat.c:6:3: warning: initialization makes pointer from integer without a cast [enabled by default]
14 stringcat.c:6:3: warning: (near initialization for ‘ap’) [enabled by default]
15 stringcat.c:6:3: warning: excess elements in scalar initializer [enabled by default]
16 stringcat.c:6:3: warning: (near initialization for ‘ap’) [enabled by default]
17 */
int (*ap)[2]=int(*)[2]a;
18 /*stringcat.c:7:16: error: expected expression before ‘int’
20 */
int a[2]={1,2}
int (*ap)[2]=(int(*)[2])a;//注意,此处申明的不是一个数组而是一个指针
注意,这个才是正确的,但是你能看出上面声明的是一个指针而不是一个数组吗?
指针和数组之所以复杂,主要原因就再次:他们的声明和使用形似并不相同。来看看C语言的声明是如何形成的?
声明器就是标识符以及与它组合在一起的任何指针,函数括号和数组下标等。
构造声明要遵循以下条件:
函数返回值不能是一个函数:foo()()
函数返回值不能是一个数组:foo()[]
数组里面不能有函数:foo[]()
但可以这样:
函数可以返回函数指针:int (*fun())()
函数可以返回指向数组的指针:int(*foo())[]
数组里面允许函数指针:int(*foo[])()
数组里面允许其他数组:int foo[][]
这里面,比较qipa的有三种类别,数组,函数,指针。
一般定义:int a;
数组:int a[5];
函数:int fun();
指针:int *p;
尤其是数组和函数,我们可以认为他们的变量处在类型修饰符(不过把运算符也算作类型的话)的中间。
我们接下来看看C语言的优先级规则
A:声明从名字开始读取,然后按照优先级次序依次开始读取
B:优先级从高到低的次序是
B1:声明中被括号扩起来的部分
B2:后缀操作符() []
B3:前缀操作符*
C:const和volatile关键字后面紧跟类型说明符的话(int,long),那么它作用与类型说明符号;反之,它作用与左边挨着的*
如此,可以分析char * const * (*next)()
具体的分析过程按照上述即可,结果是:next是一个指针,它指向一个函数,这个函数返回另一个指针,该指针是指向类型为char的常量指针。
然后再来分析以往我们提出的例子就简单多了:int (*ap)[2],ap是一个指针,这个指针指向长度为2的int类型的数组;int *ap[2],ap是一个长度为2的数组,数组中的每个元素是指向int类型的指针。这样就明晰了数组指针和指针数组的区别。
相关文章推荐
- C语言中的声明解析规则——数组,指针与函数
- 关于C语言声明、指针、数组、函数、typedef等等的一通“超级扯”
- c语言复杂变量声明(数组、指针、函数)
- c语言复杂变量声明(数组、指针、函数)
- 数组&指针&函数(转自C语言习题与解析 第3版 李春葆编著)
- 函数内 声明一个字符数组和一个指向字符串数组的指针 区别
- C语言中指针,数组,字符串及函数
- C语言之指针、数组和函数
- c函数指针的优先级规则---怎么来读一个很长的声明呢? [转]
- C语言之数组专题:数组指针、指针数组、数组做函数参数退化、数组名、数组类型
- C语言 内存分配 地址 指针 数组 参数 实例解析
- 20170215C语言提升08_指针_02数组指针及指针数组及函数指针
- C语言之指针、数组和函数
- 详解C语言中的函数、数组与指针
- 一.数组和指针的差别二.函数的好处以及使用函数的注意项。三.用c语言如何实现面向对象?c语言哪些点体现了面向对象。
- 【嵌入式开发】C语言 结构体相关 的 函数 指针 数组
- 20170215C语言提升08_指针_02数组指针及指针数组及函数指针
- C语言之指针、数组和函数
- C语言指针系列 - 一级指针.一维数组,二级指针,二维数组,指针数组,数组指针,函数指针,指针函数
- C语言之指针、数组和函数