当前位置: 首页 > news >正文

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

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


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

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

资源泄漏场景

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

编译器可能的执行顺序

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

风险分析

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

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

安全初始化模式

// 正确:独立语句完成资源分配和智能指针构造
std::shared_ptr<Widget> pw = std::make_shared<Widget>(); // 推荐方式// 或显式new+智能指针构造(同一语句)
std::shared_ptr<Widget> pw(new Widget); // 安全构造processWidget(pw, riskyFunction()); // 异常安全调用

现代C++最佳实践

// 优先使用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++标准允许编译器重排函数参数求值顺序(未指定顺序)

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

自定义RAII类的安全用法

// 自定义资源管理类
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与智能指针构造
    • 禁用多步操作初始化智能指针

错误用法重现

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

安全重构方案

// 方案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());
http://www.dtcms.com/a/311773.html

相关文章:

  • Python 程序设计讲义(50):Python 的可迭代对象与迭代器
  • Flutter基础知识
  • SpringBoot与TurboGears2跨栈、整合AI服务、智能客服路由系统整合实战
  • SpringCloud学习第一季-4
  • 第15届蓝桥杯Python青少组中/高级组选拔赛(STEMA)2024年3月10日真题
  • 17、原坐标变换和逆变换在实战中用法
  • 无人机数字图传技术的前沿探索与应用
  • 【昇腾推理PaddleOCR】生产级部署方式
  • 机器学习实战:KNN算法全解析 - 从原理到创新应用
  • LangChain框架入门05:输出解析器使用技巧
  • SpringBoot 服务器配置
  • Json简单的实现
  • 【Android】RecyclerView实现新闻列表布局(1)适配器使用相关问题
  • 【Leetcode】2561. 重排水果
  • 【Django】-6- 登录用户身份鉴权
  • 知识随记-----Qt 实战教程:使用 QNetworkAccessManager 发送 HTTP POST
  • 面试小总结
  • 解决技术问题思路
  • STM32学习记录--Day6
  • Spring 中 Bean 的生命周期
  • 知识蒸馏 - 基于KL散度的知识蒸馏 HelloWorld 示例
  • Linux网络编程【UDP网络通信demon】
  • 网页操作自动化解决方案:如何用Browser-Use+CPolar提升企业运营效率
  • React ahooks——副作用类hooks之useThrottleFn
  • 【智能体cooragent】新智能体创建相关代码解析
  • 双网卡UDP广播通信机制详解
  • 海洋大地测量基准与水下导航系列之九我国海洋PNT最新技术进展(下)
  • P13014 [GESP202506 五级] 最大公因数
  • 使用WSL2开发zephyr
  • 重型机械作业误伤预警响应时间缩短80%!陌讯多模态识别算法在工程现场的应用优化