C/C++:如何理解复杂的声明

简介:

http://blog.chinaunix.net/u/12783/showart_378340.html

 

C/C++:如何理解复杂的声明

这里说的声明,不光适用于C/C++,其他的一些语言也能适用。

与java和C#等不同,声明和定义在C/C++中有着比较明显的区别:声明仅仅是介绍名字(introduce names),而定义则会为该名字分配相应的空间。打个通俗的比喻:声明就是你在谈话中提到某个人的名字,而定义就是把你提到的这个人带到谈话的人群中来,让大家见识一下他/她是什么样子。

这里主要介绍声明。
在C中,声明的形式为(dcl是declaration的简写):
dcl: optional *'s direct-dcl(含有可选"*"的direct-dcl)
direct-dcl name
                (dcl)
                direct-dcl()
                direct-dcl[optional size] 
根据该规则进行逆向解析,就可以得到正确的声明。简化一下:“TypeName Declarator;”其中,Declarator就是声明中的那个名字。当你遇到任何你不能理解的声明时,这个法则就是救命稻草。最简单的例子:
int aInt;
这里,int是TypeName,aInt是Declarator。
再说明一下结合紧密度。在声明/定义变量时,可以使用一些修饰比如“*”,“[]”,“()”等。“()”(非函数声明中的“()”)具有最高的紧密度,其次才是函数和数组的“()”和“[]”。
没有“*”的声明称为直接声明(direct-dcl),而有“*”称为声明(dcl)。直接声明要比声明结合的紧。分解声明时,先读出结合紧的。在这里,我把direct-dcl称为更紧的结合,它比dcl结合得紧。
最后,需要你用英语来读出这个声明。对于“[]”,应该读成array of。
对于复杂的定义,可以将其分解。比如“T (*p)()”可以分解成“T D1()”,D1读作:function returning T。其中D1是*p。那么该声明应该读成:p is a poniter to。二者合在一起,就变成了p is a pointer to function returning T,即:p是指向返回T类对象的函数的指针。

再看一个稍微复杂的示例:
T (*pfa[])();
根据dcl和direct-dcl,可以分解成T1 D1(因为结合紧密度),T1, 也就是T (),那么应该读作:
D1 is function returning T。
D1又可以写成T2 D2,其中T2是T1 [],可以分解成T1 D2[],读作:
array of D2 function returning T。
D2是指针,读作:pointers to。那么整个“T (*pfa[])();”应该读作:
pfa is an array of pointers to function returning T,即:pfa是个存放指向返回T类对象函数的指针的数组。

换种方式看,在这个例子中,pfa是名字,T(*[])()是类型。将(*pfa[])视为一体(direct-dcl),称为D1,那么可以写成T D1(),function returning object of T。在D1中,将*pfa视为一体(dcl),称为D2,那么*pfa[]应该是D2[](direct-dcl),array of D2。合起来就是array of D2 function returning object of T。D2是*pfa(dcl),替换到前面这句话,结果就是array of pointers to function returning object of T。

有了这些说明,可以试着做一下下面的题,看看自己是否真的理解了:
char **argv
argv: pointer to pointer to char
int (*daytab)[13]
daytab: pointer to array[13] of int
int *daytab[13]
daytab: array[13] of pointer to int
void *comp()
comp: function returning pointer to void
void (*comp)()
comp: pointer to function returning void
char (*(*x())[])()
x: function returning pointer to array[] of
pointer to function returning char
char (*(*x[3])())[5]
x: array[3] of pointer to function returning
pointer to array[5] of char
有了这个,就很容易理解下面这两个typedef:
typedef void (*disp)(int);
typedef void (*signal(int, disp))(int);

在C++中,规则比C要复杂一些。不过,基本思想保持不变,按照C的原则来理解复杂的声明,基本上就能满足要求了。没有在这里列出C++的规则一方面是因为太广,不能覆盖全;另一个原因就是,按照C的规则来就足够了,毕竟C++要与C兼容。

这里讨论的仅仅是声明,不涉及到类型的signature,因此相对来说还是比较简单的。

参考:
The C programming Language, by Brian W. Kernighan and Dennis M. Ritchie
The C++ programming Language, by Bjarne Stroustup
 
Copyleft (C) 2007-2009 raof01.
本文可以用于除商业外的所有用途。此处“用途”包括(但不限于)拷贝/翻译(部分或全部),不包括根据本文描述来产生代码及思想。若用于非商业,请保留此 权利声明,并标明文章原始地址和作者信息;若要用于商业,请与作者联系(raof01@gmail.com),否则作者将使用法律来保证权利。
目录
相关文章
|
3月前
|
编译器 Shell Linux
C语言的本质(六):链接详解-定义和声明
C语言的本质(六):链接详解-定义和声明
46 0
|
1月前
|
编译器 C++
在C++语言中类的定义和声明
在C++语言中类的定义和声明
15 0
|
7月前
|
数据安全/隐私保护 C++
【C++学习手札】一文带你初识C++类和对象(空间大小、类内外声明、成员函数)
【C++学习手札】一文带你初识C++类和对象(空间大小、类内外声明、成员函数)
|
7月前
常量引用的概念
常量引用的概念
24 0
|
11月前
|
程序员 C语言 C++
C++ 命名空间、域、缺省参数、函数重载、引用、auto、内联函数的知识点+完整思维导图+基本练习题+深入细节+通俗易懂建议收藏(一)
C++ 命名空间、域、缺省参数、函数重载、引用、auto、内联函数的知识点+完整思维导图+基本练习题+深入细节+通俗易懂建议收藏(一)
|
11月前
|
安全 编译器 C语言
C++ 命名空间、域、缺省参数、函数重载、引用、auto、内联函数的知识点+完整思维导图+基本练习题+深入细节+通俗易懂建议收藏(二)
C++ 命名空间、域、缺省参数、函数重载、引用、auto、内联函数的知识点+完整思维导图+基本练习题+深入细节+通俗易懂建议收藏(二)
|
网络协议 测试技术 Go
结构体声明和使用陷阱|学习笔记
快速学习结构体声明和使用陷阱
85 0
结构体声明和使用陷阱|学习笔记
|
前端开发
前端工作总结169-声明问题 的定义变量
前端工作总结169-声明问题 的定义变量
57 0
前端工作总结169-声明问题 的定义变量
和鸭哥一起看,Java数组声明及初始化的全貌
和鸭哥一起看,Java数组声明及初始化的全貌
105 0
|
程序员 数据库
自然框架的声明
1、 自然框架是由控件、类库、思路构成的,他不是生成器!   2、 自然框架不是平台。   3、 自然框架不想干掉写代码的程序员,而是想把大家从繁复的代码中解脱出来,把更多的精力放在更重要的地方,比如处理复杂的业务逻辑。
645 0

热门文章

最新文章