码迷,mamicode.com
首页 > 编程语言 > 详细

C++编程技巧减少编译时间

时间:2014-10-23 12:34:25      阅读:231      评论:0      收藏:0      [点我收藏+]

标签:前置声明

1. #define的保护
所有头文件都应该使用#define 防止头文件被多重包含(multiple inclusion),命名格式
当是:<PROJECT>_<PATH>_<FILE>_H_
为保证唯一性,头文件的命名应基于其所在项目源代码树的全路径。例如,项目foo 中的头
文件foo/src/bar/baz.h 按如下方式保护:
#ifndef FOO_BAR_BAZ_H_
#define FOO_BAR_BAZ_H_
...

#endif // FOO_BAR_BAZ_H_

2. 头文件依赖
使用前置声明(forward declarations)尽量减少.h 文件中#include 的数量。
当一个头文件被包含的同时也引入了一项新的依赖(dependency),只要该头文件被修改,
代码就要重新编译。如果你的头文件包含了其他头文件,这些头文件的任何改变也将导致那
些包含了你的头文件的代码重新编译。因此,我们宁可尽量少包含头文件,尤其是那些包含
在其他头文件中的。
使用前置声明可以显著减少需要包含的头文件数量。举例说明:头文件中用到类File,但不
需要访问File 的声明,则头文件中只需前置声明class File;无需#include
"file/base/file.h"。
在头文件如何做到使用类Foo 而无需访问类的定义?
1) 将数据成员类型声明为Foo *或Foo &;
2) 参数、返回值类型为Foo 的函数只是声明(但不定义实现);
3) 静态数据成员的类型可以被声明为Foo,因为静态数据成员的定义在类定义之外。
另一方面,如果你的类是Foo 的子类,或者含有类型为Foo 的非静态数据成员,则必须为
之包含头文件。
有时,使用指针成员(pointer members,如果是scoped_ptr 更好)替代对象成员(object
members)的确更有意义。然而,这样的做法会降低代码可读性及执行效率。如果仅仅为
了少包含头文件,还是不要这样替代的好。

当然,.cc 文件无论如何都需要所使用类的定义部分,自然也就会包含若干头文件。
译者注:能依赖声明的就不要依赖定义。


以上是Google中C++编程规范的前两条要求。第一条相信大家都知道。第二条就是我要说的问题了。

这里我用VS2012解释下这个事情。要减少编译的目的就是要减少#include;

那我们先定义这样两个类:

#pragma once
class A
{
public:
	A(void);
	~A(void);
};

#pragma once
#include "A.h"
class B
{
public:
	B(void);
	~B(void);
private:
	A a;
};

可以看到这是很简单的组合类的情况。B类中有个A的对象。下面我们生成一下。

bubuko.com,布布扣

这时候A,B第一次编译成功

接下来我们修改对象A,在对象A中增加一个私有成员i,B不变;

#pragma once
class A
{
public:
	A(void);
	~A(void);
private:
	int i;
};

这时候再生成一次,可以看到如下情况

bubuko.com,布布扣


可以看到A编译后。编译器发现B中导入了A.h然后又编译了一次B

在工程量比较小的时候这么做是没关系的。但当你编译一次代码需要1分钟2分钟的时候。我们频繁的修改源码然后编译你会崩溃的是不是?

所以前置声明是一个可以选择的解决方案。

B的代码只要做如下修改,取消导入A.h在B.h中声明class A;

#pragma once
class A;
class B
{
public:
	B(void);
	~B(void);
private
	A* a;
};

我们同样在A没修改之前做一次编译。

bubuko.com,布布扣

同样我们在A中加入私有成员后再编译一次。

bubuko.com,布布扣

可以看到这里只编译了A.cpp文件

这种原因就是其实是C++的声明与定义并未完全分离的缘故。像Java就不会有这种问题=。=




C++编程技巧减少编译时间

标签:前置声明

原文地址:http://blog.csdn.net/leejuen/article/details/40392643

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!