(C++17) 未捕获异常 uncaught_exceptions

文章目录

🐾前言

从 C++11 其有 std::uncaught_exception 用于判断++当前线程++ 是否有未捕获的异常。而这个函数只是返回 bool 仅能查看是否存在未捕获异常。

因此到了 C++17 被弃用,在 C++20 中移除。C++17 引入了 int std::uncaught_exceptions() 可以获得具体的数量。

std::uncaught_exception, std::uncaught_exceptions - cppreference.com

🐾实例

🪝工具类

其实这个非常简单,最常用的就是在析构函数的时候,进行获取判断。

因此可以写这么一个工具类。当然,是选择在判断出有异常还是没异常的时候触发这个 action 还是两种状态下都注册一个,都是可以的。

cpp 复制代码
#include <exception>
#include <functional>

class ExcCatch {
private:
    const int             count = 0;
    std::function<void()> action;

public:
    ExcCatch(const std::function<void()>& action = {})
        : count(std::uncaught_exceptions()), action(action) {}

    ~ExcCatch() {
        if (count != std::uncaught_exceptions()) {
            if (action) {
                action();
            }
        }
    }
};

🪝测试

此处是用 vscode 中的 Copilot 所生成的测试代码。

简单看看使用方式就行了。

cpp 复制代码
#include <exception>
#include <functional>

class ExcCatch {
private:
    const int             count = 0;
    std::function<void()> action;

public:
    ExcCatch(const std::function<void()>& action = {})
        : count(std::uncaught_exceptions()), action(action) {}

    ~ExcCatch() {
        if (count != std::uncaught_exceptions()) {
            if (action) {
                action();
            }
        }
    }
};

#include <atomic>
#include <iostream>
#include <stdexcept>

// Helper for test output
#define ASSERT_TRUE(cond, msg)                      \
    if (!(cond)) {                                  \
        std::cout << "[FAIL] " << msg << std::endl; \
    } else {                                        \
        std::cout << "[PASS] " << msg << std::endl; \
    }
#define ASSERT_FALSE(cond, msg) ASSERT_TRUE(!(cond), msg)

// Test that ExpCatch does not call action if no exception is thrown
void test_NoException_ActionNotCalled() {
    std::atomic<bool> called{false};
    {
        ExcCatch guard([&] { called = true; });
    }
    ASSERT_FALSE(called, "NoException_ActionNotCalled");
}

// Test that ExpCatch calls action if exception is thrown
void test_Exception_ActionCalled() {
    std::atomic<bool> called{false};
    try {
        ExcCatch guard([&] { called = true; });
        throw std::runtime_error("test");
    } catch (...) {
        // Exception is caught here
    }
    ASSERT_TRUE(called, "Exception_ActionCalled");
}

// Test that ExpCatch works with empty action
void test_EmptyAction_NoCrash() {
    try {
        ExcCatch guard;
        throw std::runtime_error("test");
    } catch (...) {
        // Should not crash
    }
    std::cout << "[PASS] EmptyAction_NoCrash" << std::endl;
}

// Test that ExpCatch does not call action if exception is thrown but not caught in scope
void test_ExceptionNotCaughtInScope_ActionCalled() {
    std::atomic<bool> called{false};
    auto              test_func = [&] {
        ExcCatch guard([&] { called = true; });
        throw std::runtime_error("test");
    };
    try {
        test_func();
    } catch (...) {
        // Exception is caught here
    }
    ASSERT_TRUE(called, "ExceptionNotCaughtInScope_ActionCalled");
}

int main() {
    test_NoException_ActionNotCalled();
    test_Exception_ActionCalled();
    test_EmptyAction_NoCrash();
    test_ExceptionNotCaughtInScope_ActionCalled();
    return 0;
}

⭐END

🌟交流方式

关注我,学习更多C/C++,python,算法,软件工程,计算机知识!

⭐交流方式⭐ |C/C++|算法|设计模式|软件架构-CSDN社区
B站

👨‍💻主页:天赐细莲 bilibili

相关推荐
SunnyDays10115 分钟前
Python操作Excel批注:从基础添加到高级自定义的完整指南
开发语言·python·excel
Yyyyyy~14 分钟前
【C++】数组篇
开发语言·c++
qq_3331209718 分钟前
C++高并发内存池的整体设计和实现思路_C 语言
java·c语言·c++
牛肉在哪里18 分钟前
ros2 从零开始27 编写广播C++
开发语言·c++·机器人
Curvatureflight25 分钟前
前端国际化 i18n 落地实践:语言包、动态文案和格式化问题怎么处理?
前端·c++·vue
yong999033 分钟前
基于Qt的文件传输系统
开发语言·qt
yuan1999733 分钟前
基于 MATLAB PSO 工具箱的函数寻优算法
开发语言·算法·matlab
黄小白的进阶之路42 分钟前
C++提高编程---3.9 STL-常用容器-map/multimap 容器【P231~P235】
c++
WBluuue1 小时前
Codeforces 1096 Div3(ABCDEFGH)
c++·算法
誰能久伴不乏1 小时前
ibmodbus “Invalid argument“ 错误的排查与修复
c++·qt·modbus