编程风格指南系列一

373次阅读  |  发布于2年以前

我们在写代码的时候,应该保证代码有以下特性:

  1. 正确性,语法正确,结果正确
  2. 可读性,通用的,必须的习惯用语和模式可以让代码更加容易理解。
  3. 可维护性,程序应对变化的能力,容易优化。

因此,一种好的编程规范可以让团队的合作效率提高,这里推荐Google的编程 Google C++ style guide

感谢译者Yang.Y和 YuleFox做出的贡献

接下来进入正题:

Part11. 头文件

通常每一个 .cc文件都有一个对应的 .h文件.也有一些常见例外,如单元测试代码和只包含 main()函数的 .cc文件.正确使用头文件可令代码在可读性、文件大小和性能上大为改观. 下面的规则将引导你规避使用头文件时的各种陷阱.

1.1. Self-contained头文件

Tip: 头文件应该能够自给自足(self-contained,也就是可以作为第一个头文件被引入),以 .h结尾。至 于用来插入文本的文件,说到底它们并不是头文件,所以应以 .inc结尾。不允许分离出 -inl.h头文件 的做法.

所有头文件要能够自给自足。换言之,用户和重构工具不需要为特别场合而包含额外的头文件。详言之,一个头文件要有#define保护,统统包含它所需要的其它头文件,也不要求定义任何特别 symbols.不过有一个例外,即一个文件并不是 self-contained的,而是作为文本插入到代码某处。或者,文件内容实际上是其它头文件的特定平台(platform-specific)扩展部分。这些文件就要用 .inc文件扩展名。如果 .h文件声明了一个模板或内联函数,同时也在该文件加以定义。凡是有用到这些的 .cc文件,就得统统包含该头文件,否则程序可能会在构建中链接失败。不要把这些定义放到分离的 -inl.h文件里(译者注:过去该规范曾提倡把定义放到 -inl.h里过)。

有个例外:如果某函数模板为所有相关模板参数显式实例化,或本身就是某类的一个私有成员,那么它就只能定义在实例化该模板的 .cc文件里。

1.2. #define保护

Tip: 所有头文件都应该使用 #define 来防止头文件被多重包含, 命名格式当是:H

为保证唯一性,头文件的命名应该基于所在项目源代码树的全路径. 例如,项目 foo中的头文件 foo/ src/bar/baz.h可按如下方式保护:

#ifndef FOO_BAR_BAZ_H_
#define FOO_BAR_BAZ_H_
...
#endif // FOO_BAR_BAZ_H_

1.3.前置声明

Tip: 尽可能地避免使用前置声明。使用 #include包含需要的头文件即可。

定义:所谓「前置声明」(forward declaration)是类、函数和模板的纯粹声明,没伴随着其定义.优点:

缺点:

// b.h:
struct B {};
struct D : B {};

// good_user.cc:
#include "b.h"
void f(B*);
void f(void*);
void test(D* x) { f(x); } // calls f(B*)

如果 #include被 B和 D的前置声明替代,test()就会调用 f(void*) .

结论

至于什么时候包含头文件,参见 # include的路径及顺序。

1.4.内联函数

Tip: 只有当函数只有 10行甚至更少时才将其定义为内联函数.

定义:当函数被声明为内联函数之后,编译器会将其内联展开,而不是按通常的函数调用机制进行调用.优点:只要内联的函数体较小,内联该函数可以令目标代码更加高效.对于存取函数以及其它函数体比较短,性能关键的函数,鼓励使用内联.缺点:滥用内联将导致程序变得更慢. 内联可能使目标代码量或增或减,这取决于内联函数的大小.内联非常短小的存取函数通常会减少代码大小,但内联一个相当大的函数将戏剧性的增加代码大小.现代处理器由于更好的利用了指令缓存,小巧的代码往往执行更快。结论: 一个较为合理的经验准则是,不要内联超过 10行的函数. 谨慎对待析构函数,析构函数往往比其表面看起来要更长,因为有隐含的成员和基类析构函数被调用!另一个实用的经验准则:内联那些包含循环或 switch语句的函数常常是得不偿失 (除非在大多数情况下,这些循环或 switch语句从不被执行).

有些函数即使声明为内联的也不一定会被编译器内联,这点很重要;比如虚函数和递归函数就不会被正常内联.通常,递归函数不应该声明成内联函数.(YuleFox注:递归调用堆栈的展开并不像循环那么简单,比如递归层数在编译时可能是未知的,大多数编译器都不支持内联递归函数).虚函数内联的主要原因则是想把它的函数体放在类定义内,为了图个方便,抑或是当作文档描述其行为,比如精短的存取函数.

1.5. #include的路径及顺序

Tip: 使用标准的头文件包含顺序可增强可读性,避免隐藏依赖:相关头文件, C库, C++库,其他库的 .h,本项目内的 .h.

项目内头文件应按照项目源代码目录树结构排列,避免使用 UNIX特殊的快捷目录 . (当前目录)或 ..(上级目录).例如, google-awesome-project/src/base/logging.h应该按如下方式包含:

#include "base/logging.h"

又如, dir/foo.cc或 dir/foo_test.cc的主要作用是实现或测试 dir2/foo2.h的功能, foo.cc中包含头文件的次序如下:

  1. dir2/foo2.h (优先位置,详情如下)
  2. C系统文件
  3. C++系统文件
  4. 其他库的 .h文件
  5. 本项目内 .h文件

这种优先的顺序排序保证当 dir2/foo2.h遗漏某些必要的库时,dir/foo.cc或 dir/foo_test.cc的构建会立刻中止。因此这一条规则保证维护这些文件的人们首先看到构建中止的消息而不是维护其他包的人们。

dir/foo.cc 和 dir2/foo2.h 通常位于同一目录下 (如 base/basictypes_unittest.cc 和 base/ basictypes.h),但也可以放在不同目录下.

按字母顺序分别对每种类型的头文件进行二次排序是不错的主意。注意较老的代码可不符合这条规则,要在方便的时候改正它们。

您所依赖的符号 (symbols)被哪些头文件所定义,您就应该包含(include)哪些头文件,前置声明 (forward declarations)情况除外。比如您要用到 bar.h中的某个符号,哪怕您所包含的 foo.h已经包含了 bar.h, 也照样得包含 bar.h,除非 foo.h有明确说明它会自动向您提供 bar.h中的 symbol.不过,凡是 cc文件所对应的「相关头文件」已经包含的,就不用再重复包含进其 cc文件里面了,就像 foo.cc只包含 foo.h就够了,不用再管后者所包含的其它内容。

举例来说, google-awesome-project/src/foo/internal/fooserver.cc的包含次序如下:

#include "foo/public/fooserver.h" //优先位置

#include <sys/types.h>
#include <unistd.h>

#include <hash_map>
#include <vector>

#include "base/basictypes.h"
#include "base/commandlineflags.h"
#include "foo/public/bar.h"

例外:有时,平台特定(system-specific)代码需要条件编译(conditional includes),这些代码可以放到其它 includes之后。当然,您的平台特定代码也要够简练且独立,比如:

#include "foo/public/fooserver.h"

#include "base/port.h" // For LANG_CXX11.

#ifdef LANG_CXX11
#include <initializer_list>
#endif // LANG_CXX11

note:

  1. 避免多重包含是学编程时最基本的要求;
  2. 前置声明是为了降低编译依赖,防止修改一个头文件引发多米诺效应;
  3. 内联函数的合理使用可提高代码执行效率;
  4. -inl.h可提高代码可读性 (一般用不到);
  5. 标准化函数参数顺序可以提高可读性和易维护性 (对函数参数的堆栈空间有轻微影响,我以前大多是相同类型放在一起);
  6. 包含文件的名称使用 .和 ..虽然方便却易混乱,使用比较完整的项目路径看上去很清晰,很条理,包含文的次序除了美观之外,最重要的是可以减少隐藏依赖,使每个头文件在“最需要编译”(对应源文件处:D)的地方编译,有人提出库文件放在最后,这样出错先是项目内的文件,头文件都放在对应源文件的最前面,这一点足以保证内部错误的及时发现了.
  7. 还真有项目用 #includes来插入文本,且其文件扩展名 .inc看上去也很科学。
  8. Google已经不再提倡 -inl.h用法。
  9. 注意,前置声明的类是不完全类型(incomplete type),我们只能定义指向该类型的指针或引用,或者声明(但不能定义)以不完全类型作为参数或者返回类型的函数。毕竟编译器不知道不完全类型的定义,我们不能创建其类的任何对象,也不能声明成类内部的数据成员。
  10. 类内部的函数一般会自动内联。所以某函数一旦不需要内联,其定义就不要再放在头文件里,而是放到对应的 .cc文件里。这样可以保持头文件的类相当精炼,也很好地贯彻了声明与定义分离的原则。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8