c++怎么处理try-catch异常_c++异常捕获与错误处理机制详解

C++中try-catch用于异常处理,由try、catch、throw组成,支持多类型捕获与自定义异常,结合RAII和noexcept可提升程序健壮性与资源安全性。

在C++中,try-catch机制是标准的异常处理方式,用于捕获和响应程序运行期间可能发生的错误。合理使用异常处理可以提升程序的健壮性和可维护性。

1. 基本语法:try-catch结构

C++中的异常处理由try、catch、throw三个关键字组成。

基本结构如下:

try {
    // 可能抛出异常的代码
    throw exception_type("错误信息");
}
catch (exception_type& e) {
    // 捕获并处理特定类型的异常
    std::cout << "捕获异常:" << e.what() << std::endl;
}

示例:

#include 
#include 

int main() { try { throw std::runtime_error("测试异常"); } catch (const std::runtime_error& e) { std::cout << "运行时错误:" << e.what() << std::endl; } return 0; }

2. 多种异常类型的捕获

一个try块可以对应多个catch块,按类型分别处理异常。

注意:catch的顺序应从派生类到基类,避免被覆盖。

try {
    // ...
    throw std::invalid_argument("参数错误");
}
catch (const std::invalid_argument& e) {
    std::cout << "参数异常:" << e.what() << std::endl;
}
catch (const std::runtime_error& e) {
    std::cout << "运行时异常:" << e.what() << std::endl;
}
catch (const std::exception& e) {
    std::cout << "通用异常:" << e.what() << std::endl;
}
catch (...) {
    std::cout << "未知异常被捕获" << std::endl;
}

catch(...) 可以捕获所有未被前面catch处理的异常,常用于兜底处理。

3. 抛出异常:throw的使用

throw用于主动抛出异常,支持任何可复制的类型,但推荐使用标准异常类或自定义异常类。

  • 标准异常位于头文件中,如:logic_error、runtime_error、invalid_argument
  • 自定义异常类通常继承自std::exception或其子类

自定义异常示例:

class MyException : public std::exception {
public:
    const char* what() const noexcept override {
        return "自定义异常发生";
    }
};

// 使用 throw MyException();

4. 异常安全与资源管理

异常可能导致资源泄漏,如内存、文件句柄等未释放。C++推荐使用RAII(资源获取即初始化)原则来确保异常安全。

  • 使用智能指针(如std::unique_ptr)自动管理堆内存
  • 使用std::fstream代替原始文件操作,析构时自动关闭
  • 避免在构造函数中抛出异常,若必须,则确保对象状态一致

示例:

void risky_function() {
    std::unique_ptr data(new int[1000]); // 自动释放
    if (/* 错误条件 */)
        throw std::runtime_error("出错了");
    // 即使抛出异常,data也会被自动释放
}

5. noexcept关键字与异常规范

从C++11起,noexcept用于声明函数不会抛出异常。

  • 帮助编译器优化代码
  • 某些STL操作(如vector扩容)会检查元素类型是否noexcept

用法:

void safe_func() noexcept {
    // 保证不抛出异常
}

void may_throw() noexcept(false) { throw 1; }

基本上就这些。C++的异常机制灵活但需谨慎使用。关键在于:正确分类异常、优先使用标准异常、结合RAII管理资源、避免在性能敏感路径频繁抛异常。掌握这些,就能写出更稳定可靠的C++程序。