Effective C++ 条款17:以独立语句将newed对象置入智能指针

Effective C++ 条款17:以独立语句将newed对象置入智能指针


核心思想 :使用智能指针管理动态分配的对象时,必须确保new操作与智能指针构造在同一独立语句中完成,避免编译器优化顺序导致的内存泄漏。

⚠️ 1. 跨语句初始化的危险性

资源泄漏场景

cpp 复制代码
// 危险:跨语句初始化智能指针
processWidget(std::shared_ptr<Widget>(new Widget), riskyFunction());

编译器可能的执行顺序

  1. 执行 new Widget(分配内存)
  2. 调用 riskyFunction()(可能抛出异常)
  3. 构造 shared_ptr(管理资源)

风险分析

  • riskyFunction() 抛出异常 → 步骤1分配的Widget对象内存泄漏
  • shared_ptr 尚未接管资源 → 无自动释放机制

🚨 2. 解决方案:独立语句初始化

安全初始化模式

cpp 复制代码
// 正确:独立语句完成资源分配和智能指针构造
std::shared_ptr<Widget> pw = std::make_shared<Widget>(); // 推荐方式

// 或显式new+智能指针构造(同一语句)
std::shared_ptr<Widget> pw(new Widget); // 安全构造

processWidget(pw, riskyFunction()); // 异常安全调用

现代C++最佳实践

cpp 复制代码
// 优先使用std::make_shared(C++11+)
auto pw = std::make_shared<Widget>(); 

// 需自定义删除器时:
auto pw = std::shared_ptr<Widget>(new Widget, customDeleter);

⚖️ 3. 关键原则与注意事项
原则 说明 违反后果
单语句构造原则 new操作与智能指针构造必须在同一独立语句完成 资源泄漏风险
优先make_shared/make_unique 使用工厂函数保证异常安全(C++11/14) 消除显式new
避免函数参数内构造 禁止在函数调用参数列表内直接new+智能指针构造 编译器重排风险
扩展至所有资源管理类 规则同样适用于自定义RAII类 通用资源安全原则

编译器优化陷阱

C++标准允许编译器重排函数参数求值顺序(未指定顺序)

cpp 复制代码
// 编译器可能的重排顺序:
1. new Widget        // 分配资源
2. riskyFunction()   // 可能抛出异常
3. shared_ptr构造    // 未执行 → 资源泄漏

自定义RAII类的安全用法

cpp 复制代码
// 自定义资源管理类
class DBConnection { /* ... */ };
class DBHandler {
public:
    explicit DBHandler(DBConnection* conn) : conn_(conn) {}
    ~DBHandler() { conn_->close(); }
private:
    DBConnection* conn_;
};

// 安全初始化:
DBConnection* dbc = new DBConnection;  // 危险:分离语句
DBHandler handler(dbc);                 // 可能泄漏

// 正确:同一语句完成
DBHandler handler(new DBConnection);    // 异常安全

💡 关键原则总结

  1. 异常安全第一原则
    • 动态资源必须立即被资源管理对象接管
    • new操作与RAII对象构造必须原子化完成
  2. make函数优先原则
    • std::make_shared<>()(C++11)
    • std::make_unique<>()(C++14)
    • 避免显式new操作
  3. 禁用复杂参数表达式
    • 禁止在函数调用参数内组合new与智能指针构造
    • 禁用多步操作初始化智能指针

错误用法重现

cpp 复制代码
// 危险:可能泄漏资源的写法
processResource(
    std::unique_ptr<Resource>(new Resource), // 风险点
    loadConfig() // 可能抛出异常
);

安全重构方案

cpp 复制代码
// 方案1:独立语句构造(基础)
auto res = std::unique_ptr<Resource>(new Resource);
processResource(std::move(res), loadConfig());

// 方案2:make_unique(推荐,C++14+)
auto res = std::make_unique<Resource>();
processResource(std::move(res), loadConfig());

// 方案3:延迟调用(异常安全封装)
auto callProcess = [](auto&&... args) {
    auto res = std::make_unique<Resource>();
    processResource(std::move(res), std::forward<decltype(args)>(args)...);
};
callProcess(loadConfig());
相关推荐
KuaCpp15 分钟前
C++面向对象(速过复习版)
开发语言·c++
智者知已应修善业3 小时前
【51单片机不用数组动态数码管显示字符和LED流水灯】2023-10-3
c++·经验分享·笔记·算法·51单片机
AI进化营-智能译站4 小时前
ROS2 C++开发系列16-智能指针管理传感器句柄|告别ROS2节点内存泄漏与野指针
java·c++·算法·ai
报错小能手4 小时前
好好讲讲移动构造 移动赋值
c++
syker4 小时前
AIFerric深度学习框架:自研全栈AI基础设施的技术全景
开发语言·c++
xvhao20135 小时前
单源、多源最短路
数据结构·c++·算法·深度优先·动态规划·图论·图搜索算法
笑鸿的学习笔记6 小时前
qt-C++语法笔记之Qt Graphics View 框架中的类型辨析完全指南
c++·笔记·qt
山居秋暝LS6 小时前
安装C++版opencv和opencv_contrib
开发语言·c++·opencv
谭欣辰7 小时前
LCS(最长公共子序列)详解
开发语言·c++·算法
Cando学算法7 小时前
鸽笼原理(抽屉原理)
c++·算法·学习方法