异常处理

905 阅读4分钟

异常处理

异常是一个程序执行过程中出现的问题。C++ 异常是对程序运行过程中产生的例外情况作出的响应,比如试图除以零。

异常提供一种方法将程序控制从一个程序的一部分转移到另一部分。C++ 异常处理是建立在三个关键词: 尝试,捕获和抛出之上的。

  • throw: 程序运行出现问题时抛出异常。这是使用一个 throw 关键字实现的。
  • catch: 程序用异常处理器在你想要处理问题的地方捕获异常。catch 关键字显示异常的捕获。
  • try: 一个 try 块标识一个可能会产生异常的代码块。紧随其后的是一个或多个 catch 块。

假设一个代码块将产生一个异常,结合使用 try 和 catch 关键词的方法捕获了一个异常。一个 try / catch 块放置在可能生成一个异常的代码周围。在一个 try / catch 块里面的代码被称为保护代码,try / catch 的语法规则如下:

try {
  // protected code
} catch(ExceptionName e1) {
  // catch block
} catch(ExceptionName e2) {
  // catch block
} catch(ExceptionName eN) {
  // catch block
}

你可以列出多个捕捉语句捕获不同类型的异常,以防你的 try 代码块在不同的情况下产生了不止一个异常。

抛出异常

异常可以在代码块的任何地方使用抛出语句抛出。把语句的操作数确定类型的异常,可以是任何表达式,表达式的结果的类型决定了类型的异常抛出。

下面是一个例子,在除以零条件发生时,抛出异常:

double division(int a, int b) {
  if (b == 0) {
    throw "Division by zero condition!";
  }
  return (a/b);
}

捕获异常

try 块后的 catch 块可以捕获任何异常。您可以指定你需要捕获何种类型的异常,这是由出现在关键字 catch 后边的括号中的异常声明确定的。

try {
  // protected code
} catch(ExceptionName e) {
  // code to handle ExceptionName exception
}

上面的代码将捕获到一个 ExceptionName 类型的异常。如果您想要指定一个 catch 块可以应该处理任何在 try 代码中产生的异常,你必须将一个省略号…放在 catch 后的括号中,异常声明如下:

try {
  // protected code
} catch(...) {
  // code to handle any exception
}

下面是一个例子,这个例子抛出会除零异常,我们在 catch 块里面捕获它

#include <iostream>
using namespace std;

double division(int a, int b) {
  if(b == 0) {
    throw "Division by zero condition!";
  }
  return (a/b);
}

int main() {
  int x = 50;
  int y = 0;
  double z = 0;

  try {
    z = division(x, y);
    cout << z << endl;
  } catch (const char* msg) {
    cerr << msg << endl;
  }

  return 0;
}

因为上例中提出了一个 const char * 类型的异常,所以捕捉这个异常时,我们必须在 catch 块中使用 const char *。如果我们编译和运行上面的代码,这将产生以下结果:

Division by zero condition!

C++ 标准异常

C++ 在 <exception> 中提供了一系列标准的异常,我们可以用在我们的程序中。这些异常使用父-子分层结构展示如下:

img

这是对上面提到的层次结构中每个异常的描述:

  • std::exception
    • 异常和所有标准 C++ 异常的父类。
  • std::bad_alloc
    • 该异常可能会在使用 new 关键字时抛出。
  • std::bad_cast
    • 该异常可以由 dynamic_cast 抛出。
  • std::bad_exception
    • 这是一个在 C++ 程序中处理意想不到的异常的有效手段。
  • std::bad_typeid
    • 该异常可以由 typeid 抛出。
  • std::logic_error
    • 理论上可以通过阅读代码发现的异常。
  • std::domain_error
    • 这是一个在数学无效域被使用时抛出的异常。
  • std::invalid_argument
    • 参数非法时会抛出的异常。
  • std::length_error
    • 太大的 std::string 被创造时,抛出异常。
  • std::out_of_range
    • 可以抛出该异常的方法例如 std::vectorstd::bitset::operator[]()
  • std::runtime_error
    • 理论上不能通过读代码检测到的异常。
  • std::overflow_errorr
    • 如果出现数字溢出,则抛出该异常。
  • std::range_error
    • 当你试图存储一个超过范围的值的时候,会抛出该异常。
  • std::underflow_error
    • 如果出现数学下溢时,抛出该异常。

定义新异常

你可以采用继承及重写异常类来。下面是示例,显示如何使用 std::exception 类以标准的方式实现自己的异常:

#include <iostream>
#include <exception>
using namespace std;

struct MyException : public exception {
  const char * what() const throw() {
    return "C++ Exception";
  }
};

int main() {
  try {
    throw MyException();
  } catch(MyException& e) {
    std::cout << "MyException caught" << std::endl;
    std::cout << e.what() << std::endl;
  } catch(std::exception& e) {
    //Other errors
  }
}

这将产生如下的结果:

MyException caught
C++

这里,what() 是一个异常类提供的公共方法,所有子异常类都覆盖了该方法。这将返回一个异常的原因。