C++异常处理时的异常类型抛出选择

在 C++ 中选择抛出哪种异常类型,主要取决于错误的性质以及希望传达的语义信息。以下是一些指导原则,帮助在可能发生异常的地方选择合适的异常类型进行抛出:

1. std::exception

  • 适用场景:作为所有标准异常的基类,std::exception 本身通常不直接用于抛出异常,而是作为自定义异常类的基类。
  • 使用场景:如果需要定义自己的异常类,可以继承 std::exception

2. std::runtime_error

  • 适用场景:用于报告运行时错误,即在程序运行过程中由于外部条件或不可预见的情况导致的错误。
  • 常见情况
    • 文件操作失败(如文件无法打开)。
    • 网络连接失败。
    • 动态加载库失败。
    • 其他与运行环境相关的错误。

3. std::logic_error

  • 适用场景:用于报告逻辑错误,即程序内部的逻辑问题,通常是由于程序设计不当或输入不符合预期导致的错误。
  • 常见情况
    • 参数验证失败(如函数参数不符合要求)。
    • 无效的状态转换。
    • 未实现的功能被调用。
    • 其他由于程序逻辑错误导致的问题。

4. std::out_of_range

  • 适用场景:用于报告越界错误,即访问了超出有效范围的索引或位置。
  • 常见情况
    • 访问容器(如 std::vectorstd::string)时索引超出范围。
    • 访问数组时索引越界。

5. std::bad_alloc

  • 适用场景:用于报告内存分配失败的错误。
  • 常见情况
    • new 操作符无法分配内存时,会抛出 std::bad_alloc
    • 其他内存分配相关的失败情况。

具体选择的依据

  1. 错误的性质

    • 如果错误是由于运行时外部条件导致的,选择 std::runtime_error
    • 如果错误是由于程序逻辑问题导致的,选择 std::logic_error
    • 如果错误是由于越界访问导致的,选择 std::out_of_range
    • 如果错误是由于内存分配失败导致的,选择 std::bad_alloc
  2. 语义清晰性

    • 选择能够最好地描述错误类型的异常类,这样可以让捕获异常的代码更容易理解错误的来源和性质。
  3. 代码风格和团队约定

    • 在团队开发中,遵循团队的编码规范和异常处理约定,保持一致性。

示例代码

#include <iostream>
#include <stdexcept>
#include <vector>void processFile(const std::string& filename) {// 模拟文件打开失败的运行时错误if (filename.empty()) {throw std::runtime_error("Filename is empty");}
}void validateInput(int value) {// 模拟逻辑错误,参数不符合要求if (value < 0) {throw std::logic_error("Input value cannot be negative");}
}int getElement(std::vector<int>& vec, size_t index) {// 模拟越界错误if (index >= vec.size()) {throw std::out_of_range("Index out of range");}return vec[index];
}int main() {try {processFile(""); // 可能抛出 std::runtime_error} catch (const std::runtime_error& e) {std::cout << "Runtime error: " << e.what() << std::endl;}try {validateInput(-5); // 可能抛出 std::logic_error} catch (const std::logic_error& e) {std::cout << "Logic error: " << e.what() << std::endl;}try {std::vector<int> vec = {1, 2, 3};getElement(vec, 5); // 可能抛出 std::out_of_range} catch (const std::out_of_range& e) {std::cout << "Out of range error: " << e.what() << std::endl;}return 0;
}

选择抛出哪种异常类型,主要依据错误的性质和希望传达的语义信息。std::runtime_error 用于运行时错误,std::logic_error 用于逻辑错误,std::out_of_range 用于越界错误,std::bad_alloc 用于内存分配错误。通过合理选择异常类型,可以使代码更具可读性和可维护性。

在C++中,选择抛出哪种标准异常类需要根据错误的类型、发生场景以及标准库的规范来判断。具体的选择依据和常见场景如下:

一、标准异常类的分类与适用场景

  1. std::exception
    • 基类:所有标准异常的基类,通常不直接抛出,而是通过其派生类使用。
    • 适用场景:当需要统一捕获所有异常时,例如:
      catch (const std::exception& e) {std::cerr << e.what();  // 输出错误信息
      }
      
  2. std::runtime_error
    • 运行时错误:表示无法在编译时检测到的错误,例如内存分配失败、系统资源不足等。
    • 典型用例:
      • 内存分配失败时抛出 std::bad_allocruntime_error 的子类)。
      • 文件操作失败(如打开不存在的文件)。
  3. std::logic_error
    • 逻辑错误:表示可以通过代码逻辑避免的错误,例如参数校验失败、算法逻辑错误等。
    • 常见子类:
      • std::invalid_argument:参数无效(如除零操作)。
      • std::out_of_range:索引越界(如访问容器超出范围的元素)。
      • std::domain_error:数学运算中使用无效的输入域(如对负数开平方根)。
  4. 其他具体异常类
    • std::bad_castdynamic_cast 类型转换失败时抛出。
    • std::bad_typeidtypeid 操作符作用于 NULL 指针时抛出。

二、选择异常类的判断依据

  1. 错误类型是否可预见
    • 逻辑错误(logic_error):若错误可通过代码逻辑提前检测(如参数校验),应抛出逻辑错误类。
    • 运行时错误(runtime_error):若错误无法在编译时或运行时早期检测(如内存分配失败),应抛出运行时错误类。
  2. 错误的粒度
    • 优先使用具体子类:例如,参数无效时应抛出 std::invalid_argument,而非更宽泛的 std::runtime_error
    • 自定义异常类:若标准类无法准确描述错误,可继承 std::exception 自定义异常类。
  3. 标准库的约定
    • 遵循标准库的异常抛出规则。例如:
      • std::vector::at() 在越界时抛出 std::out_of_range
      • new 操作符在内存不足时抛出 std::bad_alloc

三、示例代码

  1. 参数校验失败(逻辑错误)
    void divide(int a, int b) {if (b == 0) {throw std::invalid_argument("Divisor cannot be zero");}
    }
    
  2. 内存分配失败(运行时错误)
    int* createArray(size_t size) {try {return new int[size];} catch (const std::bad_alloc& e) {std::cerr << "Memory allocation failed: " << e.what();return nullptr;}
    }
    
  3. 索引越界(逻辑错误)
    std::vector vec = {1, 2, 3};
    try {int value = vec.at(5);  // 抛出 std::out_of_range
    } catch (const std::out_of_range& e) {std::cerr << "Index out of range: " << e.what();
    }
    

四、最佳实践

  1. 优先使用标准异常类:避免重复造轮子,提高代码可读性和可维护性。
  2. 明确异常边界:在函数文档中注明可能抛出的异常类型。
  3. 避免过度捕获:尽量捕获具体异常类型,而非笼统的 catch (...),以提高错误处理的精确性。
    通过以上规则和示例,可以更合理地选择抛出异常的类型,使代码更加健壮和易维护。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HC-05与HC-06蓝牙配对零基础教程 以及openmv识别及远程传输项目的概述

这个是上一年的项目&#xff0c;之前弄得不怎么完整&#xff0c;只有一个openmv的&#xff0c;所以openmv自己去我主页找&#xff0c;这篇主要讲蓝牙 这个是我在使用openmv连接单片机1然后单片机1与单片机2通过蓝牙进行通信 最终实现的效果是&#xff1a;openmv识别到图形和数…

点云分割方法

点云分割 通过判断三维距离&#xff0c;实现对创建3团点云的分割 通过判断三维距离&#xff0c;实现对创建3团点云的分割 * 点云1 gen_object_model_3d_from_points (rand(100), rand(100),rand(100), Points1)* 点云2 gen_object_model_3d_from_points (rand(100), 2rand(100…

SpringBoot3使用CompletableFuture时java.util.ConcurrentModificationException异常解决方案

问题描述 在Spring Boot 3项目中&#xff0c;使用CompletableFuture进行异步编程时&#xff0c;偶发{"code":500,"msg":"java.util.ConcurrentModificationException"}异常&#xff0c;但代码中并未直接操作List或CopyOnWriteArrayList等集合类…

细说卫星导航:测距定位原理

测距定位原理 1. 伪距测量技术 核心原理&#xff1a;卫星发射信号&#xff0c;用户接收并记录传播时间&#xff0c;乘以光速得到距离&#xff08;伪距&#xff09;。 技术细节&#xff1a; 信号传播路径分析 信号结构&#xff1a; 卫星信号包含三部分&#xff1a; 载波&…

Linux系统管理与编程09:任务驱动综合应用

兰生幽谷&#xff0c;不为莫服而不芳&#xff1b; 君子行义&#xff0c;不为莫知而止休。 [环境] windows11、centos9.9.2207、zabbix6、MobaXterm、Internet环境 [要求] zabbix6.0安装环境&#xff1a;Lamp&#xff08;linux httpd mysql8.0 php&#xff09; [步骤] 5 …

RAG(Retrieval-Augmented Generation)基建之PDF解析的“魔法”与“陷阱”

嘿&#xff0c;亲爱的算法工程师们&#xff01;今天咱们聊一聊PDF解析的那些事儿&#xff0c;简直就像是在玩一场“信息捉迷藏”游戏&#xff01;PDF文档就像是个调皮的小精灵&#xff0c;表面上看起来规规矩矩&#xff0c;但当你想要从它那里提取信息时&#xff0c;它就开始跟…

RK3568 I2C底层驱动详解

前提须知&#xff1a;I2C协议不懂的话就去看之前的内容吧&#xff0c;这个文章需要读者一定的基础。 RK3568 I2C 简介 RK3568 支持 6 个独立 I2C: I2C0、I2C1、I2C2、I2C3、I2C4、I2C5。I2C 控制器支持以下特性: ① 兼容 i2c 总线 ② AMBA APB 从接口 ③ 支持 I2C 总线主模式…

UNIX网络编程笔记:基本TCP套接字编程

一、socket函数 一、socket函数核心参数与协议组合 函数原型与基本功能 #include <sys/socket.h> int socket(int family, int type, int protocol);• 功能&#xff1a;创建通信端点&#xff08;套接字&#xff09;&#xff0c;返回描述符供后续操作。 • 返回值&#…

JSON在AutoCAD二次开发中应用场景及具体案例

配置文件的读取 在AutoCAD插件开发中&#xff0c;可能需要生成、修改、读取配置文件中一些参数或设置。JSON格式的配置文件易于编写和修改&#xff0c;且可以方便地反序列化为对象进行使用。 运行后效果如下 using Autodesk.AutoCAD.ApplicationServices; using Autodesk.Au…

自由学习记录(46)

CG语法的数据类型 // uint : 无符号整数&#xff08;32位&#xff09; // int : 有符号整数&#xff08;32位&#xff09; // float : 单精度浮点数&#xff08;32位&#xff09;&#xff0c;通常带后缀 f&#xff08;如 1.0f&#xff09; // half : 半精度浮…

解决Selenium滑动页面到指定元素,点击失效的问题

White graces&#xff1a;个人主页 &#x1f649;专栏推荐:Java入门知识&#x1f649; &#x1f439;今日诗词:君失臣兮龙为鱼&#xff0c;权归臣兮鼠变虎&#x1f439; ⛳️点赞 ☀️收藏⭐️关注&#x1f4ac;卑微小博主&#x1f64f; ⛳️点赞 ☀️收藏⭐️关注&#x1f4…

Vue基础

目录 -Vue基础- 1、插值表达式 {{}} 2、Vue核心特性&#xff1a;响应式 3、开发者工具Vue Devtools(极简插件下载) 4、Vue指令 v-text v-html v-bind v-on v-if v-show v-for v-model 5、Vue指令修饰符 .stop .prevent .capture .self .once .enter、.tab、…

收数据花式画图plt实战

目录 Python plt想把纵坐标化成对数形式代码 子图ax. 我又有ax scatter&#xff0c;又有ax plot&#xff0c;都要去对数 数字接近0&#xff0c;取对数没有定义&#xff0c;怎么办 创建数据 添加一个小的常数以避免对数未定义的问题 创建一个figure和一个子图ax 在子图a…

二项式分布(Binomial Distribution)

二项式分布&#xff08;Binomial Distribution&#xff09; 定义 让我们来看看玩板球这个例子。假设你今天赢了一场比赛&#xff0c;这表示一个成功的事件。你再比了一场&#xff0c;但你输了。如果你今天赢了一场比赛&#xff0c;但这并不表示你明天肯定会赢。我们来分配一个…

【算法工程】大模型开发之windows环境的各种安装

1. 背景 最近由于研究需要&#xff0c;我购置了两块3090显卡&#xff0c;以便在家中进行一些小规模的实验。为此&#xff0c;还更换了主机。当然&#xff0c;新系统上少不了要安装各种开发环境。从开发体验来看&#xff0c;macOS无疑更为流畅&#xff0c;但为了确保所有环境都能…

论文阅读笔记:Denoising Diffusion Probabilistic Models (2)

接论文阅读笔记&#xff1a;Denoising Diffusion Probabilistic Models (1) 3、论文推理过程 扩散模型的流程如下图所示&#xff0c;可以看出 q ( x 0 , 1 , 2 ⋯ , T − 1 , T ) q(x^{0,1,2\cdots ,T-1, T}) q(x0,1,2⋯,T−1,T)为正向加噪音过程&#xff0c; p ( x 0 , 1 , …

vscode查看文件历史git commit记录

方案一&#xff1a;GitLens 在vscode扩展商店下载GitLens 选中要查看的文件&#xff0c;vscode界面右上角点击GitLens的图标&#xff0c;选择Toggle File Blame 界面显示当前打开文件的所有修改历史记录 鼠标放到某条记录上&#xff0c;可以看到记录详情&#xff0c;选中O…

【数据挖掘】Python基础环境安装配置

【数据挖掘】Python基础环境安装配置 一、摘要二、安装Python3.13.2三、安装Jupyter Notebook四、安装Numpy和Pandas以及matplotlib五、安装scikit-learn库和seaborn库 一、摘要 本文主要介绍如何在Windows上安装Python3.13.2&#xff0c;然后基于该Python版本安装Jupyter not…

DeepSeek写打台球手机小游戏

DeepSeek写打台球手机小游戏 提问 根据提的要求&#xff0c;让DeepSeek整理的需求&#xff0c;进行提问&#xff0c;内容如下&#xff1a; 请生成一个包含以下功能的可运行移动端打台球小游戏H5文件&#xff1a; 要求 可以重新开始游戏 可以暂停游戏 有白球和其他颜色的球&am…

SpringMVC的执行流程剖析和源码跟踪

目录 一、常用组件:1、DispatcherServlet2、HandlerMapping3、Handler4、HandlerAdapter:5、ViewResolver6、View 二、SpringMVC的执行流程:1、流程图 在这里插入图片描述2、文字解析流程图3、ContextLoaderListener 三、源码跟踪1、doService()方法2、doDispatch()方法逻辑分解…