首页 > 编程语言 >【C++进阶学习】第十二弹——C++ 异常处理:深入解析与实践应用

【C++进阶学习】第十二弹——C++ 异常处理:深入解析与实践应用

时间:2024-08-09 19:53:52浏览次数:19  
标签:第十二 std 进阶 抛出 捕获 C++ catch 异常 throw

前言:

在C++编程语言中,异常处理是一种重要的机制,它允许程序员在运行时捕获和处理错误或异常情况。本文将详细介绍C++异常处理的相关知识点,包括异常的定义、抛出与捕获、异常处理的原则、以及在实际编程中的应用。

目录

1. 异常处理的基本概念

1.1 异常的定义

1.2 异常的抛出

1.3 异常的捕获

2. 异常的使用

2.1 异常抛出和匹配的原则

2.2 在函数调用链中异常栈的展开匹配原则

3. 异常的重新抛出

4. 异常安全

5. 异常规格

6. C++异常处理的实践应用

6.1 文件操作异常

6.2 数学运算异常

7. 总结


1. 异常处理的基本概念

在C++中,异常处理是一种机制,用于处理运行时发生的错误或异常情况。异常可以是程序执行过程中遇到的任何问题,如除以零、文件读写错误、资源未正确释放等。

1.1 异常的定义

在C++中,异常是一个对象,通常由std::exception或其派生类创建。异常对象包含了错误信息和状态,程序员可以使用这些信息来诊断和处理错误。

1.2 异常的抛出

异常的抛出使用throw关键字。程序员在代码中使用throw语句来抛出异常,这可以是显式抛出一个异常对象,也可以是抛出一个特定类型的异常(如std::runtime_error)。

throw std::runtime_error("发生了一个错误");
1.3 异常的捕获

异常的捕获使用try...catch块。try块包含可能抛出异常的代码,而catch块用于捕获并处理这些异常。

try {
    // 可能抛出异常的代码
} catch (const std::exception& e) {
    // 处理异常
    std::cerr << "捕获到异常: " << e.what() << std::endl;
}

一个try后面可以跟着多个catch,因为一段代码可能出现多种异常

try
{
  // 保护的标识代码
}catch( ExceptionName e1 )
{
  // catch 块
}catch( ExceptionName e2 )
{
  // catch 块
}catch( ExceptionName eN )
{
  // catch 块
}

2. 异常的使用

2.1 异常抛出和匹配的原则

1. 异常是通过抛出对象来激活的,该对象的类型决定了应该激活那个catch的处理代码

2. 如果有多个处理代码与对象类型匹配,那么就激活离的最近的一个

3. 抛出异常对象时,会生成一个临时对象的拷贝,这个临时对象的拷贝会在被catch以后销毁

4. 异常的捕获所有原则:

  • 可以使用 catch(...) 来捕获所有类型的异常。这种捕获方式通常用于那些不关心异常具体类型,只想处理所有异常的情况。
2.2 在函数调用链中异常栈的展开匹配原则

1. 首先检查throw本身是否在try块内部,如果是再查找是否有匹配的catch,如果有,则直接调用

2. 如果所在函数栈没有匹配的catch,则退出当前函数栈,到调用该函数的栈中进行寻找

3. 如果找到main函数的栈中,依然没有匹配的catch,则会直接终止程序。为了防止终止程序的这种情况出现,我们一般都会在main函数中加入一个catch(...)捕获任意类型的异常

4. 找到匹配的catch后 ,就会继续执行catch中的语句
#include<iostream>
using namespace std;
double func2(int x, int y)
{
	if (x == 0)
		throw "除0错误";
	else
		return (double)x / (double)y;
}
void func1()
{
	int x, y;
	cin >> x >> y;
	cout << func2(x, y) << endl;
}
int main()
{
	try{
		func1();
	}
	catch (const int e){
		cout << e << endl;
	}
	catch (const char* e){
		cout << e << endl;
	}
	catch (...) {
		cout << "未知异常" << endl;
	}
	return 0;
}

3. 异常的重新抛出

  • 在 catch 块中,可以使用 throw(不带参数);来重新抛出当前捕获的异常。这通常用于在处理完一些资源清理工作后,将异常传递给更高层的调用者。

void func1()
{
	// 这里可以看到如果发生除0错误抛出异常,另外下面的array没有得到释放。
    // 所以这里捕获异常后并不处理异常,异常还是交给外面处理,这里捕获了再
    // 重新抛出去。
	int* arr = new int[10];
	try{
		int x, y;
		cin >> x >> y;
		cout << func2(x, y) << endl;
	}
	catch(...){
		delete arr;
		throw;
	}
}

4. 异常安全

  • 在构造函数和析构函数中应避免抛出异常,因为这可能导致对象状态不一致或资源泄漏。
  • 应该使用 RAII(Resource Acquisition Is Initialization)原则来管理资源,确保异常发生时资源能够自动释放。(这个会在后面讲智能指针时讲到)

5. 异常规格

  • 可以在函数声明中使用异常规格来指定函数可能抛出的异常类型。这有助于调用者了解预期的异常,并做出相应的处理。

下面是几种常见的异常规格:

// 这里表示这个函数会抛出A/B/C/D中的某种类型的异常
void fun() throw(A,B,C,D);
// 这里表示这个函数只会抛出bad_alloc的异常
void* operator new (std::size_t size) throw (std::bad_alloc);
// 这里表示这个函数不会抛出异常
void* operator delete (std::size_t size, void* ptr) throw();
// C++11 中新增的noexcept,表示不会抛异常
thread() noexcept;
thread (thread&& x) noexcept;

6. C++异常处理的实践应用

6.1 文件操作异常

在进行文件操作时,可以使用异常处理来捕获和处理可能发生的错误,如文件不存在、权限问题等。

#include <fstream>
#include <iostream>

void readFile(const std::string& filename) {
    std::ifstream file(filename);
    if (!file) {
        throw std::runtime_error("无法打开文件");
    }
    // 读取文件内容
}

int main() {
    try {
        readFile("example.txt");
    } catch (const std::exception& e) {
        std::cerr << "错误: " << e.what() << std::endl;
    }
    return 0;
}
6.2 数学运算异常

在进行数学运算时,可以捕获除以零等异常情况。(上面的例子中也是这种)

#include <iostream>
#include <stdexcept>

void safeDivide(double a, double b) {
    if (b == 0) {
        throw std::runtime_error("除数不能为零");
    }
    std::cout << "结果: " << a / b << std::endl;
}

int main() {
    try {
        safeDivide(10, 0);
    } catch (const std::exception& e) {
        std::cerr << "错误: " << e.what() << std::endl;
    }
    return 0;
}

7. 总结

异常能够帮助我们快速找到错误并判断错误类型,增强我们处理错误的能力,但同时异常也会带来执行流跳跃,给我们调试等带来一些难题,但总的来说,异常还是给我们工作带来极大的便利,如何正确使用异常,是我们玩转C++的重要一步。

感谢各位大佬观看,创作不易,还请各位大佬点赞支持!!!

标签:第十二,std,进阶,抛出,捕获,C++,catch,异常,throw
From: https://blog.csdn.net/2301_80220607/article/details/141035690

相关文章

  • c++入门这一篇就够了!!!
    c++简介“c++”中的++来自于c语言中的递增运算符++,该运算符将变量加1。c++起初也叫”cwithclsss”.通过名称表明,c++是对C的扩展,因此c++是c语言的超集,这意味着任何有效的c程序都是有效的c++程序。c++程序可以使用已有的c程序库。     为什么c++不叫++c呢?因为它虽然对......
  • 【C++】模板(相关知识点讲解 + STL底层涉及的模板应用)
    目录模板是什么?模板格式模板本质函数模板格式介绍显式实例化模板参数匹配原则类模板类模板的实例化非类型模板参数模板特化——概念函数模板特化类模板的特化全特化半特化偏特化三种类特化例子(放一起比较)模板分离编译STL中比较经典的模板应用(不包含argus)......
  • 移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——4.模板
    1.泛型编程如何实现一个通用的交换函数呢?voidSwap(int&left,int&right){inttemp=left;left=right;right=temp;}voidSwap(double&left,double&right){doubletemp=left;left=right;right=temp;}voidSwap(char&left,char&right)......
  • Windows图形界面(GUI)-MFC-C/C++ - 树形视图(Tree Control) - CTreeCtrl
    公开视频-> 链接点击跳转公开课程博客首页-> ​​​链接点击跳转博客主页目录树形视图(TreeControl)-CTreeCtrl创建和初始化添加和删除项获取和设置项属性操作项项选择变化项双击项展开示例代码树形视图(TreeControl)-CTreeCtrl创建和初始化Subclas......
  • Windows图形界面(GUI)-MFC-C/C++ - 列表视图(List Control) - CListCtrl
    公开视频-> 链接点击跳转公开课程博客首页-> ​​​链接点击跳转博客主页目录列表视图(ListControl)-CListCtrl创建列表视图设置列表视图属性成员函数注意事项示例代码列表视图(ListControl)-CListCtrl创建列表视图在对话框编辑器中,从工具箱中拖拽一个Li......
  • Visual C++ 官方版下载及安装教程必装(微软常用运行库合集|DLL报错必装)
    前言MicrosoftVisualC++Redistributable(简称MSVC,VB/VC,系统运行库)是Windows操作系统应用程序的基础类型库组件。此版VisualC++运行库组件合集(微软常用运行库合集)由国内封装爱好者@Dreamcast打包而成,整合VisualC++组件安装包运行库所有版本,提供图形安装界面,可自选更新V......
  • SQL进阶技巧:有序数据合并问题之如何按照时间顺序对数据进行合并?【腾讯互娱-分析某用户
    目录0需求描述​编辑 1数据准备 2数据分析 3小结0需求描述题目:有一个流水表,用户1在什么时间玩了什么游戏。dtstatdateqqdteventtimegame2022-01-1012022-01-1000:04......
  • 《信息学奥赛一本通编程启蒙》3031-3050(Scratch、C、C++、python)
    3031:练7.3买图书(C、C++、python)3031:练7.3买图书(C、C++、python)-CSDN博客3032:练7.4梯形面积(C、C++、python)3032:练7.4梯形面积(C、C++、python)-CSDN博客3033:【例8.1】人民币支付(Scratch、C、C++、python)3033:【例8.1】人民币支付(Scratch、C、C++、python)-CSDN博客3......
  • 在国产芯片上实现YOLOv5/v8图像AI识别-【2.3】RK3588上使用C++启用多线程推理更多内容
    本专栏主要是提供一种国产化图像识别的解决方案,专栏中实现了YOLOv5/v8在国产化芯片上的使用部署,并可以实现网页端实时查看。根据自己的具体需求可以直接产品化部署使用。B站配套视频:https://www.bilibili.com/video/BV1or421T74f基础背景对于国产化芯片来说,是采用NPU进......
  • c++中的常用第三方库
    前言hello大家好,我是文宇。正文C++中常用的第三方库有很多,以下是一些常见的库,它们提供了各种功能和工具,方便开发者进行各类应用程序的开发。STL(StandardTemplateLibrary):STL是C++标准库的一部分,包含了一系列的容器、算法和函数对象等。容器包括vector(动态数组)、list(双向......