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如下: ...
随机推荐
- 遍历json数组实现树
今天小颖在工作中遇到要遍历树得问题了,实现后,怕后期遇到又忘记啦,所以记录下嘻嘻,其实这个和小颖之前写过得一篇文章 json的那些事 中第4点有关json的面试题有些类似. 数组格式: v ...
- 计算机从加电到系统(Linux)启动完成
0x0 背景 在我参加的面试和我面试别人.或者参加别人对别人的面试的事后经常遇到的一个问题就是:请从计算机加电开始描述一下计算机启动到操作系统正式启动起来的全过程.这是一个考验对计算机体系结构和基本知 ...
- 2015.7.10js-07(简单时间)
今天学习了一个小程序,将本地时间显示在页面上,用了图片的形式. 1.执行原理是,先用6张全0的图片,然后通过循环img各自根据时间来更换相对应的时间图片. 2.使用Date()函数获取本地时间,然后转 ...
- File类使用
简介 File类的实例代表了一个文件或者一个目录,通过API可以获取这个对象的相关信息. File类代表的文件或者目录可以真实存在,也可以是不存在的,可以使用File.exists()来判断. 在Wi ...
- python pytest测试框架介绍二
在介绍一中简单介绍了pytest的安装和简单使用,接下来我们就要实际了解pytest了 一.pytest的用例发现规则 pytest可以在不同的函数.包中发现用例,发现的规则如下 文件名以test_开 ...
- SDK Location not found Android Studio + Gradle
extends: http://stackoverflow.com/questions/19272127/sdk-location-not-found-android-studio-gradle I ...
- npm publish 发布
前言 我们npm publish发布的时候,一定是本地文件发布到远程仓库,并且登录到http://registry.npmjs.org(即npm adduser或npmlogin)之后,才可以进行发布 ...
- 一个js文件如何加载另外一个js文件
方法一,在调用文件的顶部加入下例代码: document.write(”<script language=javascript src=’/js/import.js’></scrip ...
- iOS - 开发屏幕及视图层次
//屏幕视图分层 .UIWindow .UILayoutContainerView .UITransitionView .UIViewControllerWrpaperView .UILayoutCo ...
- postgresql----数组类型和函数
postgresql支持数组类型,可以是基本类型,也可以是用户自定义的类型.日常中使用数组类型的机会不多,但还是可以了解一下.不像C或JAVA高级语言的数组下标从0开始,postgresql数组下标从 ...