PIMPL(一)
1 参考
- 《effective C++》 条款31:将文件间的编译关系降至最低
- PIMPL Idiom: http://c2.com/cgi/wiki?PimplIdiom
2 什么是PIMPL?
PIMPL是指pointer to implementation。通过使用指针的方式隐藏对象的实现细节。是实现“将文件间的编译依存关系降至最低”的方法之一。另一个方式是通过接口实现,但其原理一样。
PIMPL又称作“编译防火墙”、“笑脸猫技术”,它只在C/C++等编译语言中起作用。
3 为什么要使用PIMPL?
3.1 理论分析
庞大的项目,修改一个文件之后,重新编译,所有依赖该文件的文件都需要重新编译,导致编译时间太长。
3.2 工程实例
通过描述一个实例来证明上一小节的理论。
3.2.1 不使用PIMPL
文件间的依赖关系如图:
有三个源文件依赖“Person.h”,实际中可以有更多个文件依赖它,为了说明意思,我源码写的都非常简单,主要是为了表明文件间的依赖关系而已。
Person.h #ifndef PERSON_H_
#define PERSON_H_
struct Person
{
void print();
};
#endif Person.cc #include "Person.h"
#include <iostream> void
Person::print()
{
std::cout << "Person::print()" << std::endl;
} PersonUser.cc #include "Person.h" main.cc
#include "Person.h"
int main()
{
return ;
} Makefile
#
# Makefile
# author: zhaokai
# date: --
# TESTS = main all : $(TESTS) clean :
rm -f $(TESTS)
rm -f main.o PersonUser.o Person.o main.o: main.cc Person.h
g++ -c main.cc PersonUser.o : PersonUser.cc Person.h
g++ -c PersonUser.cc
Person.o: Person.cc Person.h
g++ -c Person.cc $(TESTS): main.o PersonUser.o Person.o
g++ -o main main.o PersonUser.o Person.o
现在我们开始修改Person.h文件:
#ifndef PERSON_H_
#define PERSON_H_ struct Person
{
int i; // add int i
void print();
};
#endif
然后make,结果如下:
依赖Person.h的三个文件都被重新编译了,最后链接生成执行文件。
3.2.2 使用PIMPL
使用PIMPL需要将Person类的实现移到PersonImpl类中,使用指针的方式将实现隐藏,相当于Person.h只是一个傀儡而已,而以前依赖它的文件依旧依赖之,文件间的依赖关系如图:
有三个源文件依赖“Person.h”,实际中可以有更多个文件依赖它,有两个源文件和一个头文件依赖“PersonImpl.h”。
Person.h #ifndef PERSON_H_
#define PERSON_H_
#include <memory> struct PersonImpl;
struct Person
{
void print();
private:
std::shared_ptr<PersonImpl> pImpl;
};
#endif Person.cc #include "Person.h"
#include "PersonImpl.h" void
Person::print()
{
pImpl->print();
} PersonImpl.h
#ifndef PERSONIMPL_H_
#define PERSONIMPL_H_ struct PersonImpl
{
void print();
};
#endif PersonImpl.cc #include "PersonImpl.h"
#include <iostream> void
PersonImpl::print()
{
std::cout << "PersonImpl::print()" << std::endl;
} PersonUser.cc
#include "Person.h" main.cc
#include "Person.h" int main()
{
return ;
} Makefile #
# Makefile
# author: zhaokai
# date: --
# TESTS = main all : $(TESTS) clean :
rm -f $(TESTS)
rm -f main.o PersonUser.o Person.o PersonImpl.o main.o: main.cc Person.h
g++ --std=c++ -c main.cc PersonUser.o : PersonUser.cc Person.h
g++ --std=c++ -c PersonUser.cc Person.o: Person.cc Person.h PersonImpl.h
g++ --std=c++ -c Person.cc PersonImpl.o: PersonImpl.cc PersonImpl.h
g++ --std=c++ -c PersonImpl.cc $(TESTS): main.o PersonUser.o Person.o PersonImpl.o
g++ -o main main.o PersonUser.o Person.o PersonImpl.o
现在我们开始修改PersonImpl.h文件,注意这时候Person.h已经是傀儡了,如果想给Person增加属性那应该修改PersonImpl.h文件:
#ifndef PERSONIMPL_H_
#define PERSONIMPL_H_ struct PersonImpl
{
int i; // add int i
void print();
};
#endif
然后make,结果如下:
依赖PersonImpl.h的两个文件都重新编译了,而依赖于“Person.h”的文件main.cc和PersonUser.cc都没有重新编译。
3.2.3 对比
同样是一件事情,为Person类增加属性int i;两种方法导致编译的过程就不同,我们举得例子比较小,如果有100个类似PersonUser这样的文件,那么使用PIMPL,编译时还是只有“Person.cc”和“PersonImpl”两个文件重新编译了;但是不使用PIMPL的话,就是“main.cc”,“Person.cc”和100个类似“PersonUser.cc”这样的文件重新编译,那就是102个文件。
通过上面的实例就可证明理论分析部分了。
至于如何使用PIMPL,敬请期待下一篇文章!
PIMPL(一)的更多相关文章
- pImpl
之前看代码,一直对pIml这个用法一知半解,参考这里 的一篇文章后有所收获. 总结一下,pIml的好处如下: 第一,引入更多的头文件降低编译速度.而且这个声明当然写在一个头文件里,而头文件,是不能预编 ...
- [021]转 C++ Pimpl机制
出处:http://www.cnblogs.com/gnuhpc/ 1.简介 这个机制是Private Implementation的缩写,我们常常听到诸如“不要改动你的公有接口”这样的建议,所以我们 ...
- 何为 pimpl ?
前言 你是否总因头文件包含冲突而苦恼? 你是否因头文件包含错乱而苦恼? 你是否因封装暴露了数据而苦恼? 你是否因经常改动实现而导致重新编译而苦恼? 在这里, 这些问题都不是问题, 跟随作者, 揭秘pi ...
- PIMPL设计模式的理解和使用
以下两段不同程序的比较 //file a.h #include "a.h" #include “ b.h” class A{ void Fun(); B b; } //file: ...
- 提高C++编译速度-------pimpl 模式& 桥接模式(转)
pimpl 模式(Private Implementation),我们常常听到诸如“不要改动你的公有接口”这样的建议,所以我们一般都会修改私有接口,但是这会导致包含该头文件的所有源文件都要重新编译,这 ...
- C++ 编译期封装-Pimpl技术
Pimpl技术——编译期封装 Pimpl 意思为“具体实现的指针”(Pointer to Implementation), 它通过一个私有的成员指针,将指针所指向的类的内部实现数据进行隐藏, 是隐藏实 ...
- Item 22: 当使用Pimpl机制时,在实现文件中给出特殊成员函数的实现
本文翻译自<effective modern C++>,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 如果你曾经同过久的编译时间斗争过,那么你肯定对Pi ...
- 实现私有化(Pimpl) --- QT常见的设计模式
转载自:http://blog.sina.com.cn/s/blog_667102dd0100wxbi.html 一.遇到的问题 1.隐藏实现 我们在给客户端提供接口的时候只希望能暴露它的接口,而隐藏 ...
- PIMPL(二)
文档下载 上一篇文档,PIMPL(一) 1 如何使用PIMPL 有多种方式实现PIMPL,这里按照<Effective C++>中介绍的方式. 1.1 基本步骤 假设原有Person如下: ...
随机推荐
- CentOS上传下载查看命令
之前往CentOS中上传都用ftp软件,这里介绍一种另外的上传下载方式,两个命令轻松搞定.这两个命令目前只针对Xshell和SecureCRT等远程终端软件才支持,并且还会有时间的限制.大概30秒不上 ...
- Unity3D笔记 GUI 四、实现选项卡三
一.代码: using UnityEngine; using System.Collections; /// <summary> /// 选项卡二 /// </summary> ...
- Xcode - 打开工程,提示No Scheme解决
错误提示,如图: 解决思路:
- mysql判断一个字符串是否包含某几个字符
使用locate(substr,str)函数,如果包含,返回>0的数,否则返回0
- Linux 将进程放入后台执行,解决网络,ssh断开导致进程结束(nohup, setsid, &, disown)
Linux 将进程放入后台执行,解决网络,ssh断开导致进程结束(nohup, setsid, &, disown) 1.nohup 命令 我们知道,当用户注销(logout)或者网络断开 ...
- POJ-1887 Testing the CATCHER(dp,最长下降子序列)
Testing the CATCHER Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 16515 Accepted: 6082 ...
- TCP 123=网络时间协议(NTP),Net Controller
TCP 123=网络时间协议(NTP),Net Controller
- vim编辑器的基本用法
使用linux时候,个人比较喜欢用vim编辑器,对文本进行操作. 为了方便我使用vim编辑器,特地搜索了一下教程记录于此,防止自己忘记了. 下面就是一些vim使用的基础操作: 使用vim打开软件 vi ...
- PAC 自动代理
最近看了 HTTP权威指南 里面有关于 代理的介绍,代理有很多种,今天主要来说说 自动代理PAC PAC(Proxy Auto Config) 是一个 Script:经由编写这个 Script,我们 ...
- Python 字典 dict() 函数
描述 Python 字典 dict() 函数用于创建一个新的字典,用法与 Pyhon 字典 update() 方法相似. 语法 dict() 函数函数语法: dict(key/value) 参数说明: ...