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如下: ...
随机推荐
- WPS 2019 去除自动升级 和 广告、及优化的点
搜狗输入法 里面的快捷键会影响wps的快捷键功能,需要关掉"搜狗输入法"里面的快捷键 1. 2.去除自动升级功能 3.去除 广告 WPS 2019 流程图(断网): 思维导图: 流 ...
- C# 调用存储过程出错:String[3]: Size 属性具有无效大小值 0
存储过程如下 Create PROCEDURE [dbo].[Test] @FundId int, @vchStrategyToken nvarchar(), @ErrorMessage nvarch ...
- 第二步 (仅供参考) sencha touch + PhoneGap(cordova 2.9 及其以下版本) 使用 adt eclipse进行打包
首先你得安装一个adt-eclipse 参考资料 http://www.crifan.com/android_eclipse_offline_install_adt/ 然后就可以运行adt-eclip ...
- Windows应急响应操作手册
查看表征异常 系统卡慢.宕机.CPU和内存占用高.网络拥塞或断网.磁盘空余空间无理由大幅度缩小等,根据以上表征,可以初步猜测系统面临的问题. windows 下查看系统基本信息 PS C:\Users ...
- Dockerfile创建镜像
Dockerfile是一个文本格式的配置文件,用户可以使用Dockerfile来快速创建自定义的镜像. Dockerfile由一行行命令语句组成,并且支持易#开头的注释行. 一般而言Dockerfil ...
- python开发环境搭建(python2.7.5+pyCharm2.7.3)【原创】
1.下载python 官网下载最新版python http://www.wingide.com/downloads 2.下载PyCharm 官网可下载最新版pyCharm-professional h ...
- OpenCV Cut Image via ROI 根据兴趣区域剪裁图片
我们在使用OpenCV时,有时候需要对现有图片剪裁,比如只需要一个小窗口中的内容,那么我们可以通过OpenCV的兴趣区域 Region of Interest (ROI) 来很轻易的实现,操作也很简单 ...
- java虚拟机的符号引用和直接引用
在java中,一个java类将会编译成一个class文件.在编译时,java类并不知道引用类的实际内存地址,因此只能使用符号引用来代替.比如org.simple.People类引用org.simple ...
- 树形DP总结,持续更新
自己做了动态规划的题目已经有了一个月,但是成效甚微,所以来总结一下动态规划,希望自己能够温故知新.这个博客是关于树形dp的,动态规划的一类题目. 首先从最简单的树形DP入手,树形DP顾名思义就是一棵树 ...
- 编译android --system,framework
在你的android 目录下: sudo git clone https://android.googlesource.com/platform/manifest cd manifest git b ...