「C/C++」C/C++异常处理

news2024/9/20 16:36:36

在这里插入图片描述

博客主页:何曾参静谧的博客
文章专栏:「C/C++」C/C++学习

目录

  • 相关术语
  • 一、C语言中的异常处理
    • 1.返回值来传递错误信息
    • 2.使用标准库函数对异常进行处理(不推荐)
    • 3.使用全局变量来记录错误信息(不推荐)
  • 二、C++中的异常处理
    • 1.try{}catch(){}异常处理
    • 2.C++标准的异常
    • 3.异常的重新解析(工程开发中比较实用)
    • 4.自定义异常类


相关术语

异常(Exception)处理:对程序运行时产生的异常进行处理。常见的异常如:计算异常、格式异常、内存不足异常、文件异常等。

一、C语言中的异常处理

C语言本身并没有提供内置的异常处理机制,但我们可以通过一些技巧来处理异常情况。
一般有两种方式进行处理:

  • 方法一:是通过返回值来传递错误信息
    例如在函数执行失败时返回一个特定的错误码,调用者可以根据返回值来判断函数是否执行成功并采取相应的处理方式。if()else()
  • 方法二:使用标准库函数对异常进行处理(不推荐)
    不推荐这种方法,暴力跳转导致代码可读性不高。
    assert()函数:用于在程序中检查一个条件是否成立,如果不成立,则中断程序并输出错误信息。
    setjmp()longjmp()函数:用于实现非局部跳转,在出现错误时跳转到事先设定的跳转点。
  • 方法三:是使用全局变量来记录错误信息(不推荐)
    这种方法不够安全,容易被其他代码改变这个全局变量的值,但是在一些简单的程序中可以使用。

1.返回值来传递错误信息

以下例子为一个除法的函数,当除数为0返回结果为-1

int divide(int a, int b, int* result) {
	const double eps = 1e-9;
    if (fabs(b) < eps) { // 除数为0,发生错误
        return 0; // 返回错误码0
    } else {
        *result = a / b; // 计算商
        return 1; // 返回成功码1
    }
}

2.使用标准库函数对异常进行处理(不推荐)

使用assert()函数进行处理

#include <assert.h>
#include <stdio.h>

int main() {
    int a = 10, b = 0;
    assert(b != 0); // 如果b等于0,程序会中断执行,并输出错误信息
    int result = a / b;
    printf("%d / %d = %d", a, b, result);
    return 0;
}

使用setjmp()longjmp()函数进行处理

  • int setjmp(jmp_buf env)
    将当前上下文保持在jmp_buf结构体中
  • void longjmp(jmp_buf env, int val)
    jmp_buf结构体中恢复setjmp()保存的上下文
    最终从setjmp函数调用点返回,返回值为val
#include <setjmp.h>
#include <math.h>
#include <stdio.h>

static jmp_buf env;

double divide(int a, int b) {
	const double eps = 1e-9;
	double ret = 0;
    if (fabs(b) < eps) { // 除数为0,发生错误
        longjmp(env,1);
    } else {
        ret = a / b;
    }
    return ret;
}

int main(int argc, char *argv[]){
	if(setjmp(env) == 0){
		double ret = divide(1,0);
		printf("r=%d", ret);
	}
	else{
		printf("除数为0");
	}
}

3.使用全局变量来记录错误信息(不推荐)

在多线程的环境下,全局变量可能会引起线程安全问题,因此在使用全局变量的时候需要注意线程同步问题。

二、C++中的异常处理

C++中的异常处理是一种程序控制流程的机制,用于处理程序运行时出现的异常情况,例如除以零、非法输入等等。通过使用异常处理,可以使程序在出现异常时不退出,而是能够优雅地处理异常,从而保证程序的可靠性稳定性

1.try{}catch(){}异常处理

C++中的异常处理机制主要包含以下几个关键字:

  • try:
    用于包含可能会抛出异常的代码块,对于可能抛出异常的代码需要放在try语句块中进行封装。
  • catch:
    用于捕获并处理由try块中抛出的异常,例如输出错误信息、恢复程序状态等等。可以有多个 catch 语句处理不同类型的异常。
    注意事项:
    1. 任何异常只会被catch捕捉一次。
    2. catch(...)只能在最后进行接受。
    3. 参数是严格匹配,不进行任何数据转换。
  • throw:
    用于在代码块中抛出异常,当程序在代码块中遇到了一个异常,就可以把这个异常抛出,交给try语句块之外的catch语句块进行处理。
//以下是示例代码:
#include <iostream>
#include <string>
#include <cmath>

using namespace std;

double divide(int a, int b) {
	const double eps = 1e-9;
	double ret = 0;
    if (fabs(b) < eps) { // 除数为0,发生错误
        throw 0;
    } else {
        ret = a / b;
    }
    return ret;
}

int main(int argc, char *argv[]){
	try{
		double ret = divide(1,0);
		cout << "r=" << ret << endl;
	}
	catch(...){
		cout << "除数为0" << endl;
	}
}

2.C++标准的异常

需要包含标准的头文件:#inlcude <stdexcept>

异常名描述举例
标准异常这些异常是 C++ 标准定义的异常std::runtime_error、std::logic_error、std::out_of_range
输入输出异常这些异常通常都与输入输出操作相关std::ios_base::failure 和 std::bad_cast
内存异常这些异常通常与内存分配和释放操作相关std::bad_alloc 和 std::bad_array_new_length
数值计算异常这些异常通常与数值计算相关std::overflow_error、std::underflow_error
用户自定义异常这些异常由程序员自己定义继承std::exception
#include <iostream>
#include <stdexcept>

double divide(double a, double b) {
	const double eps = 1e-9;
	double ret = 0;
    if (fabs(b) < eps) { // 除数为0,发生错误
          throw std::invalid_argument("除数为0,发生错误");
    } else {
        ret = a / b;
    }
    return ret;
}

int main(){
    double a = 10, b = 0;
    try{
        int result = divide(a, b);
        std::cout << "结果: " << result << std::endl;
    }
    catch (std::exception& e){
        std::cerr << "异常捕捉: " << e.what() << std::endl;
    }
    return 0;
}

3.异常的重新解析(工程开发中比较实用)

常用于工程开发:当调用第三方库函数时,它本身已经有自己的异常处理机制但我们想用自己的异常处理时使用异常的重新解析。

//以下是一个示例代码,演示了如何对 `open` 函数进行异常转换:
#include <iostream>
#include <stdexcept>    // 包含 std::exception 等标准异常类
#include <fstream>      // 包含 open 函数
#include <string>

class FileOpenFailed : public std::exception
{
public:
    FileOpenFailed(const std::string& fileName) : m_fileName(fileName) {}

    virtual const char* what() const noexcept{
        return ("Failed to open file: " + m_fileName).c_str();
    }

private:
    std::string m_fileName;
};

void openFile(const std::string& fileName)
{
    std::ifstream file;
    file.exceptions(std::ifstream::failbit);

    try{
        file.open(fileName);
    }catch(const std::ifstream::failure& e){// 捕获 std::ifstream::failure 异常
        throw FileOpenFailed(fileName);     // 转换为我们自己定义的异常并抛出
    }
    // 执行文件操作...
}

int main(){
    try{
        openFile("nonexistent.txt");
    }catch(const std::exception& e){// 捕获我们自己定义的异常
        std::cerr << e.what() << '';
        return -1;
    }
    // ...
    return 0;
}

4.自定义异常类

#include <iostream>
#include <exception>
#include <string>

class MyException : public std::exception
{
public:
    MyException(const char* message) : msg(message) {}
    const char* what() const throw() { return msg.c_str(); }
private:
    std::string msg;
};

void divide(int x, int y){
    if (y == 0) {
        throw MyException("Division by zero");
    }
    std::cout << "Result: " << x / y << std::endl;
}

int main(){
    try {
        divide(10, 2);
        divide(10, 0);
    } catch (MyException& e) {
        std::cout << "Exception caught: " << e.what() << std::endl;
    }
    
    return 0;
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/453865.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

内外部函数静态变量全局变量

1、函数&#xff08;封装、复用&#xff09; 功能性&#xff1a;最基本的特性&#xff1b; 扩展性&#xff1a;对于时刻变化的需求易于扩展&#xff1b; 维护性&#xff1a;对于时刻变化的需求易于维护&#xff0c;易于编码变更&#xff1b; 封装性&#xff1a;不要把所有的代…

G1回收器:区域化分代式

G1回收器概述 既然我们己经有了前面几个强大的GC,为什么还要发布Garbage First(G1)GC? 原因就在于应用程序所应对的业务越来越庞大、复杂&#xff0c;用户越来越多&#xff0c;没有GC就不能保证应用程序正常进行&#xff0c;而经常造成STW的GC又跟不上实际的需求&#xff0c…

Python 实验四 常用数据结构(1)

1.从键盘输入一个正整数列表&#xff0c;以一1结束&#xff0c;分别计算列表中奇数和偶数的和。 n int(input("请输入一个正整数&#xff1a;")) list [] while n ! -1:list.append(n)n int(input("请输入一个正整数&#xff1a;")) else:print("…

Linux下C/C++实现(网络流量分析-NTA)

网络流量分析&#xff08;NTA - Network Traffic Analysis) 就是捕捉网络中流动的数据包&#xff0c;并通过查看包内部数据以及进行相关的协议、流量、分析、统计等&#xff0c;协助发现网络运行过程中出现的问题。通过监控和分析网络环境中的流量&#xff0c;来判断流量是用在…

electron编译环境搭建和第一个桌面应用例子

前言 Electron是基于Chromium和Node.js实现的&#xff0c;所以开发人员所需要使用到的前端技术主要包括以下方面&#xff1a; 1、Html、CSS、JavaScript、ES6 2、前端开发工具Vue、Angular、React等的一种 3、其他网络、缓存、通讯、系统、跟踪等前端技术 4、对Vscode编辑…

SpringCloud --- 认识微服务、服务拆分和远程调用

一、认识微服务 随着互联网行业的发展&#xff0c;对服务的要求也越来越高&#xff0c;服务架构也从单体架构逐渐演变为现在流行的微服务架构。这些架构之间有怎样的差别呢&#xff1f; 1.1、单体架构 单体架构&#xff1a;将业务的所有功能集中在一个项目中开发&#xff0c;…

C++学习(day5)

文章目录 一. 静态成员变量和函数(static)1. 静态成员变量2. 静态成员函数 二. 类的继承&#xff08;inhert&#xff09;1. 继承2. 继承的作用3. 一个类B继承类A4. 继承格式5. 子类会继承父类的所有成员6. 当父子类中出现同名的成员时7. **继承中构造函数和析构函数调用顺序**8…

反垃圾邮件产品测试评价方法示意图

声明 本文是学习信息安全技术 反垃圾邮件产品技术要求和测试评价方法. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 反垃圾邮件产品测试评价方法 测试环境 反垃圾邮件产品的典型测试环境如图1所示。 图1 反垃圾邮件产品典型测试环境示意图 测试设…

快速识别 SLI 指标的方法:VALET

SLI&#xff0c;Service Level Indicator&#xff0c;服务等级指标&#xff0c;其实就是我们选择哪些指标来衡量我们的稳定性。而 SLO&#xff0c;Service Level Objective&#xff0c;服务等级目标&#xff0c;指的就是我们设定的稳定性目标&#xff0c;比如“几个 9”这样的目…

Android音视频开发-OpenGL ES正交投影实现方法

本文实例为大家分享了OpenGL ES正交投影展示的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下 绘制正方形 在最开始绘制的六边形里面好像看起来挺容易的&#xff0c;也没有出现什么问题&#xff0c;接下来不妨忘记前面绘制六边形的代码&#xff0c;让我们按照自己的…

设计模式——组件协作模式之模板方法模式

文章目录 前言一、“组件协作” 模式二、模板方法模式1、动机2、源码分析讲解①、结构化软件设计②、面向对象软件设计 三、模板方法模式定义四、结构要点总结 前言 一、“组件协作” 模式 现代软件专业分工之后的第一个结果是 “框架与应用程序的划分”&#xff0c;“组件协作…

部署LVS-NAT群集实验

一、 实验准备 负载调度器&#xff1a;内网关 ens33&#xff1a;192.168.109.12&#xff0c;外网关 ens37&#xff1a;12.0.0.1外网 Web节点服务器1&#xff1a;192.168.109.13 Web节点服务器2&#xff1a;192.168.109.14 NFS服务器&#xff1a;192.168.109.11 客户端&#xf…

C#基础学习--其他主题

目录 概述 字符串 使用StringBuilder类 把字符串解析为数据值 关于可空类型的更多内容 为可空类型赋值 使用空接合运算符 Main方法 文档注释 嵌套类型 析构函数和dispose模式 概述 再本章中会讲解一些重要的杂项知识 字符串 字符串是Unicode字符串数组 字符串是不可…

ISO-27145故障诊断说明

ISO-27145故障诊断说明 2.1 27145目录说明 ISO27145-1: 这里边介绍的是一般信息和用例定义&#xff1b; ISO27145-2: 这里边介绍的是与排放相关的通用数据规则&#xff0c;用于查询&#xff1b; ISO27145-3: 这里边主要介绍了支持的服务 12服务 14服务 19服务 22服务 31服务&…

【移动端网页布局】移动端网页布局基础概念 ⑧ ( 移动端页面布局方案 | 单独制作的移动端页面 - 主流 | 响应式页面兼容移动端 - 开发难度较大 )

文章目录 一、移动端页面布局方案1、单独制作的移动端页面2、响应式页面兼容移动端 一、移动端页面布局方案 移动端页面方案 : 单独制作的移动端页面 : 主流开发方案 , PC 端 与 移动端 访问的是不同的页面 , 目前的 京东 / 淘宝 等电商网站移动端页面采取的该方案 ;响应式页面…

高效编程----VSCode+ChatGPT插件

VSCode中使用ChatGPT插件 首先在VSCode中打开扩展面板&#xff0c;搜索ChatGPT&#xff0c;安装蓝色圈出插件&#xff0c;如图所示&#xff1a; 安装完成后&#xff0c;需要重启VSCode 注册账号&#xff0c;如图所示&#xff1a; 然后在ChatGPT对话框中输入信息即可使用&#…

[oeasy]python0137_相加运算_python之禅_import_this_显式转化

变量类型 相加运算 回忆上次内容 上次讲了是从键盘输入变量input 函数 可以有提示字符串需要有具体的变量接收输入的字符串 输入单个变量没有问题 但是输入两个变量之后一相加就非常离谱 怎么办呢&#xff1f;&#x1f914; 基本实验 回到 游乐场 做个 实验 两个整数 相加…

C++STL——map与set的模拟实现

map与set的模拟实现 map与set的部分源码参考改造红黑树红黑树的迭代器补全set与map的实现完整代码 map与set的部分源码参考 map和set的底层都是由红黑树实现的。 所以这里将上次实现的红黑树插入拿来用。 首先想一想&#xff0c;搜索二叉树不能修改值&#xff0c;因为会破坏整…

第十章 装饰者模式

文章目录 前言一、装饰者模式定义装饰者模式 UML图 二、装饰者模式解决星巴克咖啡订单完整代码Drink 抽象 饮料类Coffee 咖啡类继承 Drink 做一个缓冲层Espresso 意大利咖啡 继承 CoffeeLongBlack 咖啡ShortBlack 咖啡装饰者&#xff0c;调料牛奶巧克力豆浆咖啡店测试程序添加 …

01-yolo算法

要点&#xff1a; 归纳 YOLOv5 github 1 YOLO v1 1) 将一幅图像分成SxS个网格(grid cell)&#xff0c;如果某个object的中心 落在这个网格中&#xff0c;则这个网格就负责预测这个object。 2)每个网格要预测B个bounding box&#xff0c;每个bounding box 除了要预测位置之…