0

0

怎样用智能指针实现Pimpl惯用法 unique_ptr在前置声明中的应用技巧

P粉602998670

P粉602998670

发布时间:2025-08-06 11:40:02

|

337人浏览过

|

来源于php中文网

原创

使用unique_ptr实现pimpl能自动管理内存、避免资源泄漏,并需在.cpp中定义析构函数以确保看到完整类型。1.传统pimpl用原始指针手动管理内存易出错;2.用unique_ptr后,需在头文件前置声明impl并在.cpp中定义其结构,确保析构时可见完整类型;3.拷贝操作需手动实现深拷贝,移动操作可默认或显式定义;4.必要时可用unique_ptr或shared_ptr替代,但推荐优先使用unique_ptr

怎样用智能指针实现Pimpl惯用法 unique_ptr在前置声明中的应用技巧

在C++中,Pimpl(Pointer to Implementation)是一种常见的设计惯用法,用来隐藏类的实现细节、减少编译依赖。而使用

unique_ptr
来实现Pimpl,则是现代C++推荐的做法,既安全又简洁。

怎样用智能指针实现Pimpl惯用法 unique_ptr在前置声明中的应用技巧

为什么要用
unique_ptr
实现Pimpl?

传统的Pimpl写法是用原始指针指向一个不完整类型(也就是前置声明的类),然后手动在构造函数中new,在析构函数中delete。这种方式容易出错,比如忘记释放内存或者拷贝时出问题。

unique_ptr
后,自动管理内存,避免资源泄漏;而且默认生成的析构函数也能正常工作——前提是你要正确地处理前置声明和删除器的问题。

怎样用智能指针实现Pimpl惯用法 unique_ptr在前置声明中的应用技巧

如何在前置声明中正确使用
unique_ptr

这是实现Pimpl的关键一步。因为Pimpl通常会用到未定义类型的

unique_ptr
,而标准库对这种情况有要求:

  • 必须在销毁
    unique_ptr
    时能看到完整类型定义
  • 所以不能把
    unique_ptr
    成员放在头文件中的类定义里,除非你在这个翻译单元能include实现类的完整定义。

解决方法是:将类的实现细节封装进

.cpp
文件,并在类的头文件中只做前置声明。

怎样用智能指针实现Pimpl惯用法 unique_ptr在前置声明中的应用技巧

举个例子:

// widget.h
#pragma once
#include 

class Widget {
public:
    Widget();
    ~Widget();
private:
    struct Impl; // 前置声明
    std::unique_ptr pImpl;
};
// widget.cpp
#include "widget.h"
#include 

struct Widget::Impl {
    std::string name;
    int value;
};

Widget::Widget() : pImpl(std::make_unique()) {}
Widget::~Widget() = default;

这样就能安全使用

unique_ptr
了,因为析构函数是在
.cpp
文件中定义的,此时已经知道
Impl
的完整结构。


Pimpl与拷贝/移动操作的处理

使用Pimpl之后,编译器默认生成的拷贝或移动操作不会自动生效,因为

unique_ptr
是不可拷贝的。如果你需要支持这些操作,必须手动实现。

Sologo AI
Sologo AI

SologoAI 是一款AI在线LOGO生成工具,帮助用户快速创建独特且专业的品牌标识和配套VI设计。

下载

常见做法是:

  • 实现拷贝构造函数和赋值运算符,让
    pImpl
    深拷贝;
  • 移动操作可以保持默认,或者显式定义以提高可读性。

例如:

Widget(const Widget&);
Widget& operator=(const Widget&);
Widget(Widget&&) = default;
Widget& operator=(Widget&&) = default;

.cpp
中实现拷贝逻辑:

Widget::Widget(const Widget& other)
    : pImpl(std::make_unique(*other.pImpl)) {}

注意这里用了

*other.pImpl
解引用进行拷贝构造,所以要确保
Impl
也提供了合适的拷贝构造函数。


小技巧:使用
std::unique_ptr
std::shared_ptr
的情况

虽然大多数情况下用的是

std::unique_ptr
,但在某些场景下也可以考虑:

  • 如果你想用数组形式存储实现对象,可以用
    std::unique_ptr
  • 如果多个对象共享一个实现(比如共享状态),可以考虑用
    std::shared_ptr
    代替,但这时候不再是纯粹的Pimpl风格了。

不过还是建议优先用

unique_ptr
,因为它语义清晰,生命周期管理更明确。


基本上就这些。用

unique_ptr
实现Pimpl,关键在于理解前置声明和内存释放时机的关系,以及如何合理地组织代码结构。只要记住“析构时要看到完整类型”,很多问题都能迎刃而解。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1492

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

230

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

86

2025.10.17

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

274

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.12.29

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

17

2026.01.23

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

22

2026.01.23

yy漫画官方登录入口地址合集
yy漫画官方登录入口地址合集

本专题整合了yy漫画入口相关合集,阅读专题下面的文章了解更多详细内容。

91

2026.01.23

漫蛙最新入口地址汇总2026
漫蛙最新入口地址汇总2026

本专题整合了漫蛙最新入口地址大全,阅读专题下面的文章了解更多详细内容。

124

2026.01.23

热门下载

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

精品课程

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

共94课时 | 7.5万人学习

C 教程
C 教程

共75课时 | 4.2万人学习

C++教程
C++教程

共115课时 | 13.6万人学习

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

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