c++ 的代码包含头文件和实现文件两部分, 头文件一般是提供给别人(也叫客户)使用的, 不过一旦头文件发生改变,不论多小的改变,所有引用他的文件就必须重新编译,编译就要花時间,假如你做的工程比较大(比如二次封装chrome这类的开发),重新编译迅速的時间就会浪费上班的大部分時间,这样子干了一天挺累的, 不过你的老板说你没有产出,结果你被fired, 是不是很怨啊, 假如你早点看到这段文章,你就会比你的同事开发效率高那么一部分,那样被fired就不可能是你了,你说这篇文章是不是价值千金!开个玩笑 :)
言归正传,怎样介绍编译時间呢, 我知道的就3个方法:
删除不用要的#include,替代方法 使用前向声明 (forward declared )
删除不用要的一大堆私有成员变量,转而使用 ”impl” 方法
删除不用要的类之间的继承
为了讲清楚这3点,还是举个实例比较好,这样的实例我会一步一步的改进(由于我也是一点一点摸索出来了,假如哪里说错了, 你就放心的喷吧,我会和你在争论到底的,呵呵)
现在先假设你找到一个新工作,接手以前某个程序员写的类,如下
// old.h: 这便是你接收的类
//
#include <iostream>
#include <ostream>
#include <list>
// 5 个 分别是file , db, cx, deduce or error , 水平有限没有模板类
// 只用 file and cx 有虚函数.
#include "file.h" // class file
#include "db.h" // class db
#include "cx.h" // class cx
#include "deduce.h" // class deduce
#include "error.h" // class error
class old : public file, private db {
public:
old( const cx& );
db get_db( int, char* );
cx get_cx( int, cx );
cx& fun1( db );
error fun2( error );
virtual std::ostream& print( std::ostream& ) const;
private:
std::list<cx> cx_list_;
deduce deduce_d_;
};
inline std::ostream& operator<<( std::ostream& os,const old& old_val )
{ return old_val.print(os); }
这样的类看完了, 假如你已经看出了问题出在哪里, 接下来的不必看了, 你是高手, 这些基本知识对你来说太小儿科,要是像面试时被问住了愣了一下,请接着看吧
先看怎么使用先进条: 删除不用要的#include
这样的类引用 5个头文件, 那意味着那5个头文件所引用的头文件也都被引用了进来, 具体上, 不必须要引用5 个,只需引用2个就有效可以了
1.删除不用要的#include,替代方法 使用前向声明 (forward declared )
1.1删除头文件 iostream, 我刚开始学习c++ 时照着《c++ primer》 抄,只需看见关于输入,输出就把 iostream 头文件加上, 几年过去了, 现在我知道不是这样子的, 这里只是定义输出函数, 只需引用ostream 就够了
1.2.ostream头文件也不要, 替换为 iosfwd , 为什么, 原因便是, 参数和返回类型只需前向声明就可以编译通过, 在iosfwd 文件里 678行(我的环境是vs2013,不同的编译环境具体位置可能会不相同,不过都有这句声明) 有这么一句
typedef basic_ostream<char, char_traits<char> > ostream;
inline std::ostream& operator<<( std::ostream& os,const old& old_val )
{ return old_val.print(os); }
除此之外,要是你说这样的函数要操作ostream 对象, 那还是必须要#include <ostream> , 你只说对了一半, 的确, 这样的函数要操作ostream 对象, 不过请看他的函数实现,
里边没有定义一个类似 std::ostream os, 这样子的语句,话说回来,但凡出现这样子的定义语句, 就必须#include 相应的头文件了 ,由于这是请求编译器分配空间,而假如只前向声明 class XXX; 编译器怎么知道分配多大的空间给这样的对象!
看到这里, old.h头文件可以更新如下了:
// old.h: 这便是你接收的类
//
#include <iosfwd> //新替换的头文件
#include <list>
// 5 个 分别是file , db, cx, deduce or error , 水平有限没有模板类
// 只用 file and cx 有虚函数.
#include "file.h" // class file , 作为基类不能删除,删除了编译器就不知道实例化old 对象时分配多大的空间了
#include "db.h" // class db, 作为基类不能删除,同上
#include "cx.h" // class cx
#include "deduce.h" // class deduce
// error 只被用做参数和返回值类型, 用前向声明替换#include "error.h"
class error;
class old : public file, private db {
public:
old( const cx& );
db get_db( int, char* );
cx get_cx( int, cx );
cx& fun1( db );
error fun2( error );
virtual std::ostream& print( std::ostream& ) const;
private:
std::list<cx> cx_list_; // cx 是模版类型,既不是函数参数类型也不是函数返回值类型,故此cx.h 头文件不能删除
deduce deduce_d_; // deduce 是类型定义,也不删除他的头文件
};
inline std::ostream& operator<<( std::ostream& os,const old& old_val )
{ return old_val.print(os); }
到现在为止, 删除了一部分代码, 是不是心情很爽,据说看一个程序员的水平有多高, 不是看他写了多少代码,而是看他少写了多少代码。
假如你对C++ 编程有更深一步的兴趣, 接下来的文字你还是会看的,再进一步删除代码, 不过这次要另辟蹊径了
2. 删除不用要的一大堆私有成员变量,转而使用 ”impl” 方法
2.1.使用 ”impl” 实现方法写代码,减少客户端代码的编译依赖
impl 方法简单点说便是把 类的私有成员变量全部放进一个impl 类, 然后把这样的类的私有成员变量只保留一个impl* 指针,代码如下
// file old.h
class old {
//公有和保护成员
// public and protected members
private:
//私有成员, 只需任意一个的头文件发生改变或成员个数增加,减少,所有引用old.h的客户端必须重新编译
// private members; whenever these change,
// all client code must be recompiled
};
改写成这样子:
// file old.h
class old {
//公有和保护成员
// public and protected members
private:
class oldImpl* pimpl_;
// 替换原来的所有私有成员变量为这样的impl指针,指针只必须要前向声明就可以编译通过,这种写法将前向声明和定义指针放在了一起, 有效可以。
//当然,也可以离开写
// a pointer to a forward-declared class
};
// file old.cpp
struct oldImpl {
//真正的成员变量隐藏在这里, 随意改变, 客户端的代码都不必须要重新编译
// private members; fully hidden, can be
// changed at will without recompiling clients
};
不知道你看明白了没有, 看不明白请随便写个类试验下,我便是这么做的,当然凡事也都有优缺点,下边简单对比下:
使用impl 实现类 不使用impl实现类
优点 类型定义与客户端隔离, 减少#include 的次数,提高编译速度,库端的类随意修改,客户端不必须要重新编译 直接,简单明了,不必须要考虑堆分配,释放,内存泄漏问题
缺点 相对impl的指针必须使用堆分配,堆释放,時间长了会产生内存碎片,较终决定程序运行速度, 每次调用一个成员函数都要经过impl->xxx()的迅速转发 库端任意头文件发生改变,客户端都必须重新编译
改为impl实现后是这样子的:
// 只用 file and cx 有虚函数.
#include "file.h"
#include "db.h"
class cx;
class error;
class old : public file, private db {
public:
old( const cx& );
db get_db( int, char* );
cx get_cx( int, cx );
cx& fun1( db );
error fun2( error );
virtual std::ostream& print( std::ostream& ) const;
private:
class oldimpl* pimpl; //此处前向声明和定义
};
inline std::ostream& operator<<( std::ostream& os,const old& old_val )
{ return old_val.print(os); }
//implementation file old.cpp
class oldimpl{
std::list<cx> cx_list_;
deduce dudece_d_;
};
3. 删除不用要的类之间的继承
面向对象提供了继承这种机制,不过继承不要滥用, old class 的继承就属于滥用之一, class old 继承file 和 db 类, 继承file是公有继承,继承db 是私有继承
,继承file 可以理解, 由于file 中有虚函数, old 要重新定义它, 不过依据咋们的假设, 只有file 和 cx 有虚函数,私有继承db 怎么解释?! 那么先进可能的理由便是:
通过 私有继承—让某个类不能当作基类去派生其他类,类似Java里final关键字的基本功能,不过从实例看,显然没有这样的用意, 故此这样的私有继承有效不用要, 应该改用包含的方法去使用db类提供的基本功能, 这样子就可以
把”db.h”头文件删除, 把db 的实例也可以放进impl类中,较终获得的类是这样子的:
// 只用 file and cx 有虚函数.
#include "file.h"
class cx;
class error;
class db;
class old : public file {
public:
old( const cx& );
db get_db( int, char* );
cx get_cx( int, cx );
cx& fun1( db );
error fun2( error );
virtual std::ostream& print( std::ostream& ) const;
private:
class oldimpl* pimpl; //此处前向声明和定义
};
inline std::ostream& operator<<( std::ostream& os,const old& old_val )
{ return old_val.print(os); }
//implementation file old.cpp
class oldimpl{
std::list<cx> cx_list_;
deduce dudece_d_;
};
小结一下:
这篇文章只是简单的介绍了减少编译時间的几个方法:
1. 删除不用要的#include,替代方法 使用前向声明 (forward declared )
2. 删除不用要的一大堆私有成员变量,转而使用 ”impl” 方法
3. 删除不用要的类之间的继承
这几条希望对您有所赞助, 假如我哪里讲的不够清楚也可以参考附件,哪里有完整的实例,也欢迎您发表评论, 大伙一起讨论进步,哦不,加薪。 呵呵,在下篇文章我将把impl实现方法再详细剖析下,期待吧…
以上信息由新概念科技(上海网站制作,上海网站制作)为您提供,越来越多精彩信息:http://www.jianzhan0.com/
关注“新概念科技”微信公众号,迅速获取互联网较新资讯