19. 异常处理

一、什么是异常

  程序在运行过程之中,不可避免的出现一些错误,比如:使用了没有赋值的变量、使用了不存在的索引、除 0 等等。这些错误在程序中,我们称之为异常。程序运行过程中,一旦出现异常将会导致程序立即终止,异常以后的代码全部都不会执行。

二、异常的传播

  当在函数中出现异常时,如果在函数中对异常进行了处理,则异常不会继续传播。如果函数中不会对异常进行处理,,则异常会继续向函数调用处传播。如果函数调用处处理了异常,则不再传播,如果没有则继续向调用处传播。直到传递到 main() 函数中,如果依然没有处理,则程序终止,并显示异常信息。

  当程序运行过程中出现异常以后,所有的异常信息会被保存在一个专门的异常对象,而异常传播时,实际上就是异常对象抛给了调用处。

#include <iostream>
#include <vector>

using namespace std;

int main(void)
{
    vector<int> array(3, 10);
  
    array.at(4) = 100;

    return 0;
}

三、异常处理机制

  程序运行时出现异常,目的并不是让程序直接终止。C++ 是希望在出现异常时,我们可以编写代码来对异常进行处理。遇到异常时,C++ 中有两种处理机制。

3.1、try...catch机制

  在 C++ 中,提供 try...catch 语句捕获并处理异常。在使用时,把可能产生异常的代码放在 try 子句中,把处理结果放在 catch 子句中。这样,当 try 子句中的代码块出现异常时,就会执行 catch 语句块中的内容。如果 try 语句块中代码没有异常,那么 catch 语句块不会执行。这样我们就可以通过代码来处理异常,避免因为一个异常导致整个程序的终止。如果 catch 后面的括号不使用具体的异常类型,而使用 ... 代替具体异常类型,则此时它会捕获所有的异常。

try
{
    // 代码块(可能出现异常的代码)
} 
catch (异常类型 异常对象名)
{
    // 代码块(出现错误以后的处理方式)
}
catch (异常类型 异常对象名)
{
    // 代码块(出现错误以后的处理方式)
}
catch (...)
{
    // 代码块(出现错误以后的处理方式)
}
#include <iostream>
#include <vector>

using namespace std;

int main(void)
{
    vector<int> array(3, 10);
  
    try
    {
        array.at(4) = 100;
    }
    catch(exception & e)
    {
        cerr << e.what() << '\n';
    }

    cout << "Hello world" << endl;

    return 0;
}

在使用 try...catch 语句捕获异常时,当程序出现异常并处理完后,程序继续执行;

3.2、throw机制

  如果某个函数或方法可能会产生异常,但不想在当前函数或方法中处理这个异常,则可以使用 throw 语句在函数或方法中抛出异常。

throw 异常描述

  如果异常类型名为可选参数,它用于指定抛出的异常名称以及异常信息的相关描述。如果省略,就会把当前的错误原样抛出。异常描述也可以省略,如果省略,则在抛出异常时,不附带任何描述信息。

#include <iostream>

using namespace std;

int main(void)
{
    int num1 = 0, num2 = 0;
  
    cout << "Enter two numbers: " << endl;
    cin >> num1 >> num2;

    try
    {
  
        if (num2 == 0)
        {
            throw runtime_error("Division by zero");
        }
        cout << num1 / num2 << endl;
    }
    catch(exception & e)
    {
        cerr << e.what() << '\n';
    }

    return 0;
}

四、C++标准异常

  C++ 提供了一个 exception 头文件,它里面定义了 exception 类,C++ 可以把它用作其它异常类的基本。代码可以引发 exception 异常,也可以将 exception 类用作基类。

  头文件 stdexcept 定了了其它几个异常类。该文件定义了 logic_error 和 runtime_error 类,它们都是以公有方式从 exception 派生而来的额。这些类的构造函数接受一个 string 对象作为参数,该参数提供了方法 what() 以 C-风格字符串方式返回的字符数据。login_error 异常表明存在可以通过编程修复的问题,而 runtime_error 系列异常描述了可能在运行期间发生但难以预计和防范的错误。

  异常 invalid_argument 指出给函数传递了一个意外的值。异常 length_error 用来指出没有足够的空间来执行所需的操作。异常 out_of_bands 通常用于指示索引错误。

  一般而言,不同的变量具有不同的存储范围。如果计算的结果比这个范围的最小值还小时,将会导致下溢错误,比这个范围最大值还大的话,将会导致上溢错误。如果计算结果可能不在函数的允许范围之内,但没有发生上溢或下溢错误,在这种情况下,可以使用 range_error 异常。

五、自定义异常类

  我们也可以自定义异常类,只需要创建一个类继承 exception 类即可。

class AgeNegativException : public exception
{
public:
    const char * what(void) const throw()
    {
        return "年龄不能为负";
    }
};
#include <iostream>

using namespace std;

class NegativException : public exception
{
public:
    const char * what(void) const throw()
    {
        return "年龄不能为负";
    }
};

int main(void)
{
    int age = 0;
  
    cout << "请输入年龄: " << endl;
    cin >> age;

    try
    {
  
        if (age < 0)
        {
            NegativException error;
            throw error;
        }
    }
    catch(NegativException &e)
    {
        cerr << e.what() << '\n';
    }

    return 0;
}

六、noexcept关键字的使用

  在 C++ 11 中,noexcept 关键字是一个异常指示符,用来告诉编译器函数不会抛出异常,编译器可以进行更多的优化。

void func() noexcept; // 表示func不会抛出异常
#include <iostream>

using namespace std;

class MyClass 
{
public:
    MyClass(void) noexcept ;

    ~MyClass(void) noexcept ;
};

MyClass::MyClass() noexcept 
{
    std::cout << "Constructor called" << std::endl;
}

MyClass::~MyClass(void) noexcept 
{
    cout << "Destructor called" << endl;
}

int main(void) 
{
    MyClass obj;
  
    return 0;
}
posted @ 2023-05-04 19:45  星光樱梦  阅读(12)  评论(0编辑  收藏  举报