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

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.xdnf.cn/news/16855.html

相关文章:

  • 农田通量计算方法与应用;高精度感热/潜热通量反演与绘图等;农田蒸散发与能量平衡
  • 50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | QuizApp(交互式在线测验应用组件)
  • Mujoco(MuJoCo,全称Multi - Joint dynamics with Contact)一种高性能的物理引擎
  • 基于Postman进行http的请求和响应
  • linux基本系统服务——DNS服务
  • 【嵌入式汇编基础】-ARM架构基础(三)
  • 宝塔配置文件缺失导致无法正常启动
  • Java 集合框架: LinkedHashSet
  • 进程 Vs 线程
  • 【OpenGL】LearnOpenGL学习笔记01 - 环境配置、窗口创建
  • Flask + YARA-Python*实现文件扫描功能
  • 开源列式分布式数据库clickhouse
  • 深入 Go 底层原理(十三):interface 的内部表示与动态派发
  • Redisson高并发实战:Netty IO线程免遭阻塞的守护指南
  • 算法提升之数学(快速幂+逆元求法)
  • 【20min 急速入门】使用Demucs进行音轨分离
  • Redis7 String类型数据
  • 【iOS】KVO
  • MyBatisPlus之CRUD接口(IService与BaseMapper)
  • 28Rsync免密传输与定时备份
  • 关于Web前端安全防御XSS攻防的几点考虑
  • Spring Boot 全 YAML 配置 Liquibase 教程
  • C++之vector类的代码及其逻辑详解 (中)
  • DockerFile文件执行docker bulid自动构建镜像
  • CMake指令:mark_as_advanced
  • Python序列去重高级指南:保持顺序的高效去重技术
  • 错误: 找不到或无法加载主类 原因: java.lang.ClassNotFoundException
  • 云原生三剑客:Kubernetes + Docker + Spring Cloud 实战指南与深度整合
  • 分类任务当中常见指标 F1分数、recall、准确率分别是什么含义
  • 类似 Pixso 但更侧重「网页 / 软件界面设计」「前后端可视化开发」的工具