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

250 篇文章 169 订阅






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




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";
    }
}

标准异常类的继承关系如下图所示 :
在这里插入图片描述

上图中 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 . . .

在这里插入图片描述

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
【为什么还需要学习C++?】 你是否接触很多语言,但从来没有了解过编程语言的本质?你是否想成为一名资深开发人员,想开发别人做不了的高性能程序?你是否经常想要窥探大型企业级开发工程的思路,但苦于没有基础只能望洋兴叹? 那么C++就是你个人能力提升,职业之路进阶的不二之选。【课程特色】 1.课程共19大章节,239课时内容,涵盖数据结构、函数、、指针、标准库全部知识体系。2.带你从知识与思想的层面从0构建C++知识框架,分析大型项目实践思路,为你打下坚实的基础。3.李宁老师结合4大国外顶级C++著作的精华为大家推出的《征服C++11》课程。【学完后我将达到什么水平?】 1.对C++的各个知识能够熟练配置、开发、部署;2.吊打一切关于C++的笔试面试题;3.面向物联网的“嵌入式”和面向大型化的“分布式”开发,掌握职业钥匙,把握行业先机。【面向人群】 1.希望一站式快速入门的C++初学者; 2.希望快速学习 C++、掌握编程要义、修炼内功的开发者; 3.有志于挑战更高级的开发项目,成为资深开发的工程师。 【课程设计】 本课程包含3大模块基础篇本篇主要讲解c++的基础概念,包含数据型、运算符等基本语法,数组、指针、字符串等基本词法,循环、函数、等基本句法等。进阶篇本篇主要讲解编程中常用的一些技能,包含的高级技术、继承、编译链接和命名空间等。提升篇:本篇可以帮助学员更加高效的进行c++开发,其中包含型转换、文件操作、异常处理、代码重用等内容。
在 C 中,异常处理是通过错误码或者信号来实现的。常见的错误码有 errno 和返回值。当函数执行失败时,会返回一个非零值或者特定的错误码,需要在程序中进行判断和处理。 errno 是一个全局变量,用于保存最近一次函数调用的错误码。可以通过 <errno.h> 头文件中的宏来获取具体的错误码和对应的错误信息。例如: ```c #include <stdio.h> #include <errno.h> #include <string.h> int main() { FILE *fp = fopen("non_existent_file.txt", "r"); if (fp == NULL) { printf("Failed to open file: %s\n", strerror(errno)); return 1; } // do something with the file fclose(fp); return 0; } ``` 上面的代码中,当打开一个不存在的文件时,fopen 函数会返回一个空指针,并设置 errno 为 ENOENT 错误码,表示文件不存在。通过 strerror 函数将该错误码转换为对应的错误信息,并输出到控制台。 除了 errno 外,还有一些函数会返回特定的错误码,例如 malloc 函数在分配内存失败时会返回 NULL,并设置 errno 为 ENOMEM 错误码,表示内存不足。这时候需要根据返回的错误码来进行相应的处理。 另外,C 中还可以通过信号来处理异常。可以使用 signal 函数来注册信号处理函数,当产生某个信号时,会调用该函数进行处理。例如: ```c #include <stdio.h> #include <signal.h> void handle_signal(int sig) { printf("Received signal %d\n", sig); } int main() { signal(SIGINT, handle_signal); // 注册 SIGINT 信号处理函数 while (1) { // 不断循环,等待信号 // do something } return 0; } ``` 上面的代码中,通过 signal 函数注册了一个 SIGINT 信号处理函数,当用户按下 Ctrl+C 时,会产生 SIGINT 信号,并调用该函数进行处理。在信号处理函数中,可以执行一些特定的操作,例如释放资源、保存数据等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值