【C++】异常处理 ⑧ ( 标准异常类 | 标准异常类继承结构 | 常用的标准异常类 | 自定义异常类继承 std::exception 基类 )
  K1I6GvxBEuyy 2023年12月10日 20 0



文章目录

  • 一、抛出 / 捕获 多个类型异常对象
  • 1、标准异常类
  • 2、标准异常类继承结构
  • 3、常用的标准异常类
  • 二、自定义异常类继承 std::exception 基类
  • 1、自定义异常类继承 std::exception 基类
  • 2、完整代码示例 - 自定义异常类继承 std::exception 基类








一、抛出 / 捕获 多个类型异常对象




1、标准异常类



在 C++ 语言中 , 提供了一系列的 " 标准异常类 " ,

这些 " 标准异常类 " 都继承了 std::exception 基类 ,

在 标准库 中 , 抛出的异常 , 都是 标准异常类 , 都是 std::exception 类的子类 ;



2、标准异常类继承结构



标准异常类 定义在 std 命名空间 , 标准异常类 基类 std::exception 定义在 <exception> 头文件中 ;

#include <exception>

标准异常类 基类 std::exception 中提供了 what() 函数 , 用于获取异常报错信息 , what 函数的原型如下 :

namespace std {

#pragma warning(push)
#pragma warning(disable: 4577) // 'noexcept' used with no exception handling mode specified
class exception
{
    _NODISCARD virtual char const* what() const
    {
        return _Data._What ? _Data._What : "Unknown exception";
    }
}

标准异常类的继承关系如下图所示 :

【C++】异常处理 ⑧ ( 标准异常类 | 标准异常类继承结构 | 常用的标准异常类 | 自定义异常类继承 std::exception 基类 )_异常处理


上图中 runtime_error 和 logic_error 两个重要的异常类型基类 ,

  • logic_error 是 编译时 被预先检测出来的异常 , 编程足够规范可以避免此类异常 ; logic_error 定义在了 <stdexcept> 头文件中 , 继承 exception 异常基类 ;
class logic_error : public exception { // base of all logic-error exceptions
  • runtime_error 是 运行时 不能被预先检测出的异常 ; runtime_error 定义在了 <stdexcept> 头文件中 , 继承 exception 异常基类 ;
// CLASS runtime_error
class runtime_error : public exception { // base of all runtime-error exceptions



使用标准异常类 , 使用前需要导入 <stdexcept> 头文件 ;

#include <stdexcept>



3、常用的标准异常类



常用的标准异常类如下 : std::exception 是标准异常类 基类 , 定义了 what() 函数 , 该方法返回一个指向 C 字符串的指针 , 该字符串包含了描述异常的消息 ;

  • std::bad_alloc : 当无法分配内存时 , 会抛出此异常 ;
  • std::bad_cast : 当进行类型转换时 , 如果转换失败 , 会抛出此异常 ;
  • std::bad_exception : 当异常处理程序无法处理异常时 , 会抛出此异常 ;
  • std::logic_error : 当程序中出现逻辑错误时 , 会抛出此异常 ;
  • std::out_of_range : 当访问超出有效范围的数组元素、vector 或 string 时 , 会抛出此异常 ;
  • std::length_error : 当试图创建一个超过可表示长度的容器时 , 会抛出此异常 ;
  • std::domain_error : 当计算一个数学函数的结果时 , 如果结果不在定义域内 , 会抛出此异常 ;
  • std::invalid_argument : 当一个函数接收到无效的参数时 , 会抛出此异常 ;
  • std::runtime_error : 当程序运行时发生错误时 , 会抛出此异常 ;
  • std::overflow_error : 当整数运算结果太大 , 无法表示时 , 会抛出此异常 ;
  • std::range_error : 当数学函数的结果是无限大或 NaN 时 , 会抛出此异常 ;
  • std::underflow_error : 当数值下溢 , 即数值太小而无法表示时 , 会抛出此异常 ;
  • std::system_error : 当系统调用失败时 , 会抛出此异常 ;
  • std::system_fault : 这是一个用于指示由操作系统引起的错误的异常类 ;
  • std::bad_typeid : 当试图对一个对象使用 typeid 运算符 , 而该对象没有定义 typeid 时 , 会抛出此异常 ;
  • std::bad_weak_ptr : 当使用无效的弱指针时 , 会抛出此异常 ;
  • std::exception_ptr : 这是一个可以持有异常对象的指针类型 ;
  • std::future_error : 当 future 对象的结果未能按预期准备就绪时 , 会抛出此异常 ;
  • std::invalid_promise : 当 future 对象接收到无效的 promise 时 , 会抛出此异常 ;
  • std::lock_error : 当尝试锁定一个已经被锁定的互斥量(mutex)时 , 或者当尝试解锁一个未被锁定的互斥量时 , 会抛出此异常 ;
  • std::mutex_consistent_set : 当使用 std::set_lock_state 设置一个互斥量的状态时 , 如果该状态无效 , 会抛出此异常 ;
  • std::deadlock : 当在两个或更多的线程间产生死锁时 , 会抛出此异常 ;
  • std::unexpected : 当未捕获处理函数中抛出的异常时 , 会抛出此异常 ;





二、自定义异常类继承 std::exception 基类




1、自定义异常类继承 std::exception 基类



首先 , 导入 <stdexcept> 头文件 ;

#include <stdexcept>

然后 , 自定义类继承 std::exception 类 , 通过构造函数设置异常信息 , 重写 what 函数 , 在该函数中返回异常信息 ;

// 自定义类实现标准异常类基类
class eSize : public exception {
public:
	// 构造函数设置异常信息
	eSize(const char* p)
	{
		this->m_p = p;
	}

	// 重写 what 函数
	virtual const char* what() {
		return m_p;
	}

	// 异常信息
	const char* m_p;
};

再后 , 抛出异常信息 , 都抛出 eSize 类型的自定义异常类信息 , 不再像之前一样 , 抛出多个类型的异常 ;

// 1. 在 函数 中 抛出异常
void fun(int a) {
	// 判定数字大小, 只有 60 时是合法的
	// 只要传入的参数不是 60 就需要抛出不同的异常
	if (a == 60) {
		// 合法
	}
	else if (a < 0) {
		throw eSize("参数为负数");
	}
	else if (a == 0) {
		throw eSize("参数为 0");
	}
	else if (a < 60) {
		throw eSize("参数太小");
	}
	else if (a > 60) {
		throw eSize("参数太大");
	}
}

最后 , 捕获并处理异常 , 只需要处理 eSize& 类型的异常即可 ;

// 2. 捕获并处理异常
	try
	{
		// 调用可能产生异常的函数
		fun(0);
	}
	catch (eSize& e)
	{
		const char* what = e.what();
		cout << "捕获异常 : " << what << endl;
	}
	catch (...) {
		cout << "未知异常" << endl;
	}



2、完整代码示例 - 自定义异常类继承 std::exception 基类



代码示例 :

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

// 自定义类实现标准异常类基类
class eSize : public exception {
public:
	// 构造函数设置异常信息
	eSize(const char* p)
	{
		this->m_p = p;
	}

	// 重写 what 函数
	virtual const char* what() {
		return m_p;
	}

	// 异常信息
	const char* m_p;
};

// 1. 在 函数 中 抛出异常
void fun(int a) {
	// 判定数字大小, 只有 60 时是合法的
	// 只要传入的参数不是 60 就需要抛出不同的异常
	if (a == 60) {
		// 合法
	}
	else if (a < 0) {
		throw eSize("参数为负数");
	}
	else if (a == 0) {
		throw eSize("参数为 0");
	}
	else if (a < 60) {
		throw eSize("参数太小");
	}
	else if (a > 60) {
		throw eSize("参数太大");
	}
}

int main() {

	// 2. 捕获并处理异常
	try
	{
		// 调用可能产生异常的函数
		fun(0);
	}
	catch (eSize& e)
	{
		const char* what = e.what();
		cout << "捕获异常 : " << what << endl;
	}
	catch (...) {
		cout << "未知异常" << endl;
	}

	cout << "try-catch 代码块执行完毕" << endl;

	// 控制台暂停 , 按任意键继续向后执行
	system("pause");

	return 0;
};

执行结果 :

捕获异常 : 参数为 0
try-catch 代码块执行完毕
Press any key to continue . . .

【C++】异常处理 ⑧ ( 标准异常类 | 标准异常类继承结构 | 常用的标准异常类 | 自定义异常类继承 std::exception 基类 )_c++_02


【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年12月10日 0

暂无评论

推荐阅读
  8Tw5Riv1mGFK   2024年05月01日   77   0   0 C++
  BYaHC1OPAeY4   2024年05月08日   54   0   0 C++
  yZdUbUDB8h5t   2024年04月29日   54   0   0 C++
  yZdUbUDB8h5t   2024年05月05日   41   0   0 C++
  oXKBKZoQY2lx   2024年05月17日   54   0   0 C++
K1I6GvxBEuyy
作者其他文章 更多