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),否则作者将使用法律来保证权利。
目录
相关文章
|
6月前
|
编译器 Shell Linux
C语言的本质(六):链接详解-定义和声明
C语言的本质(六):链接详解-定义和声明
134 0
|
安全 程序员 编译器
C++ 引用本质
C++ 引用本质
56 0
|
23天前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。
|
29天前
|
编译器 C语言
变量的声明与定义区别
变量的声明是指预先告知编译器变量的名称和类型,但不分配内存;而定义则是声明的同时在内存中分配空间,可以初始化。简单来说,声明是告诉编译器“有这么一个东西”,定义是“创建并使用这个东西”。
85 11
|
1月前
|
编译器
经典面试题:变量的声明和定义有什么区别
在编程领域,变量的“声明”与“定义”是经典面试题之一。声明告诉编译器一个变量的存在,但不分配内存,通常包含变量类型和名称;而定义则为变量分配内存空间,一个变量必须至少被定义一次。简而言之,声明是告知变量形式,定义则是实际创建变量并准备使用。
|
1月前
|
编译器 C语言
2.4 声明变量的4个理由
将所有变量集中声明,便于读者理解和查找,尤其当变量名具有描述性时效果更佳。若变量名不够清晰,应在注释中解释其含义,以提高代码可读性。声明变量有助于编程前规划,明确所需输入、期望输出及最佳数据表示方式,同时有助于发现潜在错误,如变量名拼写错误导致的问题。根据C99之前的规范,变量声明应置于块顶部,但C99允许按需声明,有助于避免遗漏变量赋值的情况。然而,许多编译器尚未完全支持C99标准。
39 5
|
3月前
|
编译器 程序员 C语言
【C语言篇】从零带你全面了解函数(包括隐式声明等)(下篇)
⼀般情况下,企业中我们写代码时候,代码可能⽐较多,不会将所有的代码都放在⼀个⽂件中;我们往往会根据程序的功能,将代码拆分放在多个⽂件中。
67 2
|
3月前
|
程序员 编译器 Serverless
【C语言篇】从零带你全面了解函数(包括隐式声明等)(上篇)
函数的参数部分需要交代清楚:参数个数,每个参数的类型是什么,形参的名字叫什么。
52 0
|
6月前
|
编译器 C++
在C++语言中类的定义和声明
在C++语言中类的定义和声明
50 0
|
网络协议 测试技术 Go
结构体声明和使用陷阱|学习笔记
快速学习结构体声明和使用陷阱
结构体声明和使用陷阱|学习笔记