我们经常遇到一些复杂的指针类型说明,很多初学者可能感觉有点蒙,即使是非常有经验的C/C++程序员,也会对那些比较复杂些的声明感到头疼。例如,下面这个是一个包含指针的数组,还是一个指向数组的指针呢

int *a[10];

好了,这个并不是特别复杂的声明,你可能快速而又准确的说出你的答案:一个含有10个指针的数组。

那么,来一个更具挑战性的例子,下面这货又是什么呢

int (*(*vtable)[])();

大家咋看之下是不是有点犯蒙了,又是*,又是(),又是[]的,这到底是什么呢?

这篇短文希望能够教会你用一个非常简单的技巧去读懂C/C++中类似于这样的复杂声明。

黄金法则

这个法则也叫右左法则,它是下面这样的。 从标识符开始(或者最内层的结构,如果不存在标识符的话,通常出现于函数指针),首先向右看,直到遇到 ) 括号或者结束,看到什么就说出来;然后向左看,直到遇到 ( 括号或者回到行首,看到什么就说出来。跳出一层括号,重复上述过程:右看看,说出来;左看看,说出来。直到你说出变量的类型或者返回值(针对函数指针),也就表示你把声明都读完了。 注:应用上面法则时需对[]、()、*等运算符的含义和优先级比较清楚,不熟悉的同学可以上网搜搜。 好了,用上面的规则来阅读几个C声明,先从一个简单的例子入手:

int i;

从 i 开始,你向右看,啥都没看到;然后就向左看,看到了int,说出来:i是一个int。

这个也太简单了,一看就知道了,哪还需要那么麻烦用上面的法则看来看去,好的,那么下面看一个稍微复杂点的。

int *a[3];
  • 首先从 a 开始:向右看,遇到[3];说明a“是一个包含3个元素的数组”;

  • 继续向左看,遇到的是int *,说明“数组的每个元素是指向int 类型的指针”;

根据分析过程综合起来就可得到:: a 是一个包含3个元素的数组,每个元素是一个指针,指向int。

现在加上一对括号让它看起来更怪异点儿:

int (*a)[3];

像在普通表达式中一样,括号改变了阅读/计算的顺序,下面是分析过程:

  • 还是从 a 开始:向右看,遇到括号了,改变方向

  • 向左看,遇到*说明a是一个指针,遇到(括号,跳出来,改变方向;

  • 继续向右看,遇到[3];,说明指针a是指向一个包含3个元素的数组;

  • 向左看,遇到int,说明数组a的每个元素是int。

根据分析过程综合起来就可得到:a是一个指针,指向一个包含3个元素的数组,数组的每个元素是一个int。

好,再来看看下面这个

int *foo();

很好,你说:foo是一个函数,返回一个指针,指向int。

接下来跳一步:就像我们可以定义一个指向int的指针,我们也可以像下面一样定义一个指向函数的指针

int (*foo)();

还是用上面的法则分析这个式子究竟是否满足我们的要求(一个指向函数的指针):从foo开始:向右看,遇到括号,于是向左看,遇到*说明foo是一个指针,遇到左括号,跳出来;向右看,遇到()说明“指向一个函数”;向左看,遇到int,说“函数返回int”。综合起来:foo是一个指针,指向一个函数,函数返回int。

现在让我们来看看最后一个,也是前面提到过的让你看起来蒙圈的那个

int (*(*vtable)[])();

这是一个指针,指向一个数组,数组的每个元素是一个指向返回值是int类型函数的指针你能用上面的法则分析出来吗?

如果大家有兴趣,网上很多类似的例子大家可以搜搜试着分析理解,这种复杂的声明在源码中还是比较常见的,理解了这个对于阅读源码还是很有帮助的。

参考资料: http://parrt.cs.usfca.edu/doc/how-to-read-C-declarations.html http://c-faq.com/decl/spiral.anderson.html

推荐阅读:

精心整理 | 历史干货文章目录 【福利】自己搜集的网上精品课程视频分享(上) 【数据结构与算法】 通俗易懂讲解 二叉树遍历 【数据结构与算法】 通俗易懂讲解 二叉搜索树

专注服务器后台技术栈知识总结分享

欢迎关注交流共同进步

码农有道 coding

码农有道,为您提供通俗易懂的技术文章,让技术变的更简单!