【读书笔记】Effective C++ 条款8:别让异常逃离析构函数

目录

一、数组析构时的异常问题

二、用对象管理资源时的析构异常

三、处理析构异常的三种策略

[1. 直接结束程序(abort)](#1. 直接结束程序(abort))

[2. 吞下异常(记录日志)](#2. 吞下异常(记录日志))

[3. 让客户自己解决(推荐)](#3. 让客户自己解决(推荐))

四、关于"甩锅"的讨论


一、数组析构时的异常问题

假设 vector<Widget> 中的 Widget 对象在析构时抛出异常。

C++ 规定:同时存在两个未处理的异常时,程序会终止。

由于第一个 Widget 析构抛出异常时,第二个、第三个 Widget 的析构还在继续,此时就有两个异常同时存在,因此程序会被终止。

此外,析构函数抛出异常可能导致内存释放不彻底,造成资源泄漏

结论:析构函数最好不要抛出异常。

C++ 新标准 :规定 vector 等容器在析构时不能抛出异常,否则程序直接终止。

解决方案 :使用 noexcept 关键字,声明该函数不会抛出异常。如果实际抛出了异常,会直接调用 std::terminate() 结束程序。

二、用对象管理资源时的析构异常

有时会用一个额外的 manageA 对象来管理对象 A,当 manageA 生命周期结束时,其析构函数会调用 Aclose 接口。

cpp 复制代码
class A {
public:
    static A create() {
        return A();
    }
    void close() {}  // 没人规定 close 不会抛异常
};

class manageA {
public:
    manageA() {
        _a = A::create();
    }
    ~manageA() {
        _a.close();   // 如果 close 抛异常,析构函数就会抛出异常
    }
private:
    A _a;
};

如果 close 接口可能抛出异常,那么 manageA 的析构函数就可能抛出异常,这很麻烦。

三、处理析构异常的三种策略

1. 直接结束程序(abort)

最粗暴的方法:在析构函数中捕获异常,然后调用 abort() 结束程序。

cpp 复制代码
class manageA {
public:
    manageA() {
        _a = A::create();
    }
    ~manageA() {
        try {
            _a.close();
        } catch (...) {
            abort();  // 捕获任何异常后立即终止程序
        }
    }
private:
    A _a;
};

合理性:阻止异常从析构函数中传播出去,符合 C++ 的规范要求。

2. 吞下异常(记录日志)

捕获异常但不传播,仅记录日志。

cpp 复制代码
~manageA() {
    try {
        _a.close();
    } catch (...) {
        LOG(LogLevel::DEBUG) << "error";  // 记录错误日志
    }
}

要求:采用这种方案,必须确保吞掉异常后,程序仍然能够正常运行。

有时候吞下异常的方式比 abort 结束程序更好。

3. 让客户自己解决(推荐)

最好的方式是将问题甩给程序员,而不是预设是吞掉还是结束。

cpp 复制代码
class manageA {
public:
    manageA() {
        _a = A::create();
    }
    
    // 提供 close 接口,让用户主动调用
    void close() {
        _a.close();
    }
    
    ~manageA() {
        try {
            _a.close();
        } catch (...) {
            // 什么都不做,或者记录日志
            // 但此时异常已经由用户主动调用 close 时处理过了
        }
    }
private:
    A _a;
};

设计思路

  • 提供一个 close 接口,让用户有机会主动调用

  • 如果用户调用了 close 接口,就有充足的时间去处理可能抛出的异常

  • 如果用户没有调用 close 接口,那么析构时出现异常,责任在用户------因为他本来有机会去调用 close 接口

四、关于"甩锅"的讨论

作者承认:这个行为看起来像是在甩锅给用户,可能有人认为这违反了"让接口容易正确使用"的准则。

反驳

  • 提供一个 close 接口,反而是给了用户处理异常的机会

  • 用户可以选择调用 close 来主动处理异常,也可以选择不调用

  • 不调用的后果由用户自己承担

这是一种责任分离的设计:资源管理类负责自动释放资源,但异常处理的责任交给用户主动调用接口来完成。

相关推荐
L_09071 分钟前
【C++】C++11 新特性
开发语言·c++
方也_arkling4 分钟前
【Java-Day15】API篇-ArrayList集合
java·开发语言
AI人工智能+电脑小能手5 分钟前
【大白话说Java面试题 第89题】【Mysql篇】第19题:Hash 索引和 B+ 树索引的区别?它们在使用方面的区别?
java·数据库·mysql·面试·哈希算法
我是一颗柠檬7 分钟前
【Java后端技术亮点】动态路由权限(按钮级权限),细粒度控制到按钮级别
java·开发语言·后端·状态模式
Fanfanaas9 分钟前
C++ 继承
java·开发语言·jvm·c++·学习·算法
蚰蜒螟10 分钟前
走进 Linux 内核:从 touch 命令到磁盘 inode 的完整旅程
java·linux·前端
zzqssliu14 分钟前
taocarts 跨境独立站 SEO 优化实践(多语言 + 反向海淘场景)
java·javascript·php
在繁华处22 分钟前
Java从零到熟练(十一):Spring框架入门
java·开发语言·spring
小锋java123422 分钟前
【技术专题】LangChain4j 开发Java Agent智能体 - 整合SpringBoot4
java·人工智能
十五年专注C++开发24 分钟前
cereal 库:C++ 序列化的轻量之选
开发语言·c++·序列化·反序列化·cereal