0

0

C++中的PImpl idiom是什么_C++减少编译依赖与隐藏实现细节的编程技巧

下次还敢

下次还敢

发布时间:2025-11-21 10:02:03

|

202人浏览过

|

来源于php中文网

原创

c++中的pimpl idiom是什么_c++减少编译依赖与隐藏实现细节的编程技巧

PImpl idiom(Pointer to Implementation)是一种常用的C++编程技巧,用来隐藏类的实现细节并减少编译依赖。它的核心思想是将类的具体实现移到一个独立的、不公开的结构体或类中,并通过一个指针在主类中引用它。这样,即使实现发生变化,只要接口不变,使用该类的代码就不需要重新编译。

什么是PImpl idiom

PImpl 是 "Pointer to Implementation" 的缩写,也被称为“桥接模式”的简化版本。它通过在头文件中只声明一个前向声明的类和一个指向其实现的指针,把所有私有成员变量和实现细节从头文件移到源文件中。

例如,一个普通的类可能在头文件中暴露大量包含其他类的头文件,导致依赖复杂:

// widget.h
#include 
#include 
#include "big_object.h"

class Widget { public: Widget(); void dosomething(); private: std::string name; std::vector data_; BigObject heavyobj; };

每次修改BigObject或私有成员,所有包含widget.h的文件都要重新编译。而使用PImpl后:

立即学习C++免费学习笔记(深入)”;

// widget.h
class Widget {
public:
    Widget();
    ~Widget();  // 注意:需要定义析构函数
    void do_something();
private:
    class Impl;        // 前向声明
    Impl* pImpl_;      // 指向实现
};
// widget.cpp
#include "widget.h"
#include 
#include 
#include "big_object.h"

class Widget::Impl { public: std::string name; std::vector data; BigObject heavyobj; };

Widget::Widget() : pImpl(new Impl) {} Widget::~Widget() { delete pImpl; } void Widget::dosomething() { // 使用 pImpl->... }

为什么能减少编译依赖

头文件是编译依赖的主要来源。当一个头文件被修改,所有包含它的翻译单元都必须重新编译。PImpl 把对具体类型的依赖从头文件转移到了实现文件中。

好处包括:
  • 修改私有成员时,不需要重新编译使用者代码
  • 减少头文件之间的包含关系,加快编译速度
  • 降低模块间的耦合度

如何正确实现PImpl

使用PImpl需要注意资源管理和特殊成员函数的定义。

磁力开创
磁力开创

快手推出的一站式AI视频生产平台

下载
关键点:
  • 必须显式定义析构函数,因为默认析构函数无法访问前向声明的Impl类
  • 遵循“Rule of Five”:如果定义了析构函数,通常也需要定义拷贝构造、拷贝赋值、移动构造、移动赋值,或明确删除它们
  • 现代C++推荐使用std::unique_ptr代替原始指针,避免内存泄漏

改进版本:

// widget.h
#include 

class Widget { public: Widget(); ~Widget; Widget(const Widget&); Widget& operator=(const Widget&); Widget(Widget&&); Widget& operator=(Widget&&);

void do_something();

private: class Impl; std::uniqueptr pImpl; };

std::unique_ptr可以在头文件中完成删除器的定义,因此即使Impl是前向声明,也能合法释放资源。

适用场景与注意事项

PImpl 特别适合接口稳定但实现频繁变动的类,如库开发、大型项目中的模块封装。

优点:
  • 有效隔离变化,提升编译效率
  • 增强信息隐藏,提高封装性
缺点:
  • 每次访问成员都要通过指针间接访问,轻微性能开销
  • 动态内存分配,可能影响对象布局和缓存局部性
  • 增加代码复杂度

因此,是否使用PImpl应权衡项目规模、编译时间与性能要求。

基本上就这些。PImpl idiom 是一个实用的C++惯用法,尤其在大型项目中能显著改善构建性能和模块化程度。合理使用,能让代码更健壮、更易维护。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

196

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

189

2025.07.04

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1024

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

66

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

447

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.01.19

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

465

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

13

2025.12.06

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.1万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 12.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号