【C++上层应用】1. 异常处理

文章目录

  • [【 1. C++的标准异常 】](#【 1. C++的标准异常 】)

  • [【 2. 异常转移处理 】](#【 2. 异常转移处理 】)

    • [2.1 throw 抛出异常](#2.1 throw 抛出异常)
    • [2.2 try 捕获异常](#2.2 try 捕获异常)
    • [2.3 catch 捕获异常](#2.3 catch 捕获异常)
    • [2.4 实例](#2.4 实例)
  • [【 3. 定义新的异常 】](#【 3. 定义新的异常 】)

  • 异常是程序在执行期间产生的问题,比如编译报错、链接错误等。

【 1. C++的标准异常 】

  • C++ 提供了一系列标准的异常,定义在std命名空间中,我们可以在程序中使用这些标准的异常。它们是以父子类层次结构组织起来的,如下所示:

  • 下表是对上面层次结构中出现的每个异常的说明:

异常 描述
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::vector 和 std::bitset<>::operator。
std::runtime_error 理论上不可以通过读取代码来检测到的异常。
std::overflow_error 当发生数学上溢时,会抛出该异常。
std::range_error 当尝试存储超出范围的值时,会抛出该异常。
std::underflow_error 当发生数学下溢时,会抛出该异常。

【 2. 异常转移处理 】

  • 异常提供了一种转移程序控制权的方式。C++ 异常处理涉及到三个关键字:try、catch、throw。
异常相关的关键字 作用
throw throw 关键字用于 抛出一个异常
try try 代码块中存放会抛出异常的代码,它后面通常跟着一个或多个 catch 块。
catch catch 关键字用于 捕获异常。在我们想要处理问题的地方,通过异常处理程序捕获异常。

2.1 throw 抛出异常

  • throw 关键字后面的语句可以是任意的表达式,表达式结果的类型决定了抛出的异常的类型。
  • 可以利用 throw 关键字在代码块中的任何地方抛出异常。
  • 实例:在代码除以零时抛出异常,被抛出的异常的类型是 const char *。
cpp 复制代码
double division(int a, int b)
{
   if( b == 0 )
   {
      throw "Division by zero condition!";
   }
   return (a/b);
}

2.2 try 捕获异常

  • try 块中放置可能抛出异常的代码(该代码被称为 保护代码)。
  • 捕获异常通常需要 联合使用 try 和 catch 关键字。如果 try 块在不同的情境下会抛出不同的异常,这个时候可以尝试罗列多个 catch 语句,用于捕获不同类型的异常。
  • 使用 try 和 catch 语句的语法如下所示:
cpp 复制代码
try
{
   // 保护代码
}
catch( ExceptionName e1 )
{
   // catch 块
}
catch( ExceptionName e2 )
{
   // catch 块
}
catch( ExceptionName eN )
{
   // catch 块
}

2.3 catch 捕获异常

  • catch 块跟在 try 块后面,用于捕获异常。我们可以 指定想要 catch 捕捉的异常类型,是由 catch 关键字后的括号内的异常类型声明决定的。
cpp 复制代码
try
{
   // 保护代码
}
catch( ExceptionName e )
{
  // 处理 ExceptionName 异常的代码
}
  • 上面的代码会捕获一个类型为 ExceptionName 的异常。如果我们想让 catch 块能够处理 try 块抛出的任何类型的异常,则必须在异常声明的括号内使用省略号 ...,如下所示:
cpp 复制代码
try
{
   // 保护代码
}
catch(...)
{
  // 能处理任何异常的代码
}

2.4 实例

  • 抛出一个除以零的异常,并在 catch 块中捕获该异常。
    由于我们抛出了一个类型为 const char* 类型的异常 "Division by zero condition!",因此,当捕获该异常时,我们必须在 catch 块中使用 const char*。当上面的代码被编译和执行时,它会产生下列结果:
cpp 复制代码
#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;
}

【 3. 定义新的异常 】

  • 我们可以通过 继承和重载 exception 类来定义新的异常
  • 实例:在这里,what() 是异常类提供的一个公共方法,它已被所有子异常类重载。这将返回异常产生的原因。
cpp 复制代码
#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)
  {
    //其他的错误
  }
}
相关推荐
孤寂大仙v5 分钟前
【C++】STL----list常见用法
开发语言·c++·list
咩咩大主教1 小时前
C++基于select和epoll的TCP服务器
linux·服务器·c语言·开发语言·c++·tcp/ip·io多路复用
Ylucius3 小时前
动态语言? 静态语言? ------区别何在?java,js,c,c++,python分给是静态or动态语言?
java·c语言·javascript·c++·python·学习
是店小二呀3 小时前
【C++】C++ STL探索:Priority Queue与仿函数的深入解析
开发语言·c++·后端
ephemerals__3 小时前
【c++】动态内存管理
开发语言·c++
CVer儿3 小时前
条件编译代码记录
开发语言·c++
程序猿练习生4 小时前
C++速通LeetCode简单第18题-杨辉三角(全网唯一递归法)
c++·算法·leetcode
汉字萌萌哒4 小时前
【2022 CCF 非专业级别软件能力认证第一轮(CSP-J1)入门级 C++语言试题及解析】
数据结构·c++·算法
th新港4 小时前
CCF201909_1
数据结构·c++·算法·ccf
意如流水任东西4 小时前
[C++]类和对象(上)
开发语言·c++