C++ :try 语句块和异常处理

 C++ 异常处理机制:trycatch 和 throw

异常处理是 C++ 中处理运行时错误的机制,通过 ​分离正常逻辑与错误处理 提升代码可读性和健壮性。


1. 基本结构

异常处理由三个关键字组成:

  • try:包裹可能抛出异常的代码块。
  • catch:捕获并处理特定类型的异常。
  • throw:主动抛出异常对象。
try {// 可能抛出异常的代码if (error_condition) {throw exception_object; // 抛出异常}
} catch (ExceptionType1 &e) {// 处理 ExceptionType1 类型异常
} catch (ExceptionType2 &e) {// 处理 ExceptionType2 类型异常
} catch (...) {// 捕获所有未处理的异常(不推荐滥用)
}

2. 异常处理流程

注意catch(...)捕获所有异常,但无法获取异常信息,通常用于最后兜底。

  1. 抛出异常:当throw执行时,程序立即终止当前代码块,开始查找匹配的catch块。
  2. 栈展开​(Stack Unwinding):
    • 从抛出点开始,逐层退出函数调用栈,直到找到匹配的catch块。
    • 退出时,当前作用域的局部对象会被析构(依赖RAII机制)。
  3. 匹配catch
    • catch块按声明顺序匹配异常类型。
    • 若未找到匹配的catch块,调用std::terminate()终止程序。

3. 标准异常类

C++ 标准库定义了一组异常类(需包含 <stdexcept>):

异常类说明
std::exception所有标准异常的基类
std::runtime_error运行时错误(如无效输入)
std::logic_error逻辑错误(如无效参数)
std::out_of_range访问越界(如 vector::at
std::bad_alloc内存分配失败(new 失败)

示例:使用标准异常

#include <stdexcept>
#include <vector>void checkIndex(const std::vector<int>& vec, int index) {if (index < 0 || index >= vec.size()) {throw std::out_of_range("索引越界");}
}

4. 自定义异常类

通过继承 std::exception 定义特定错误类型:

#include <exception>
#include <string>class MyException : public std::exception {
public:MyException(const std::string& msg) : msg_(msg) {}const char* what() const noexcept override {return msg_.c_str();}
private:std::string msg_;
};// 使用自定义异常
throw MyException("自定义错误消息");
noexcept 关键字
  • 声明函数不抛出异常
    void safe_function() noexcept {// 保证不会抛出异常
    }
  • 移动构造函数/析构函数:建议标记为 noexcept,避免容器操作(如 std::vector 扩容)时回退到低效拷贝。
  • noexcept 运算符:检查表达式是否可能抛出异常。
    bool is_noexcept = noexcept(safe_function()); // true
 

5. 捕获异常的最佳实践
  • 按派生类到基类顺序捕获:确保更具体的异常优先处理。
  • 通过引用捕获:避免对象切片(尤其对多态异常类)。
  • 处理已知异常:避免滥用 catch (...)
  • 优先使用标准异常类:如std::runtime_error,而非基本类型。
  • 按顺序捕获异常:先捕获派生类,再捕获基类。
  • 避免空catch:至少记录错误信息。
  • 资源管理用RAII:如std::unique_ptrstd::lock_guard
  • 限制异常使用:仅处理严重错误,避免性能损耗。
#include <iostream>
#include <fstream>
#include <stdexcept>class FileReader {
public:FileReader(const std::string& filename) : file_(filename) {if (!file_.is_open()) {throw std::runtime_error("无法打开文件: " + filename);}}// 使用RAII自动关闭文件~FileReader() noexcept = default;void read() {// 读取文件内容(可能抛出异常)}private:std::ifstream file_;
};int main() {try {FileReader reader("nonexistent.txt");reader.read();} catch (const std::runtime_error& e) {std::cerr << "错误: " << e.what() << std::endl;} catch (...) {std::cerr << "未知错误" << std::endl;}return 0;
}

6. 重新抛出异常

在 catch 块中可重新抛出当前异常(保留原始异常信息):

catch (const std::exception &e) {std::cerr << "记录错误: " << e.what() << std::endl;throw; // 重新抛出异常,供上层处理
}

7. 异常安全(Exception Safety)​

确保代码在异常发生时仍保持数据一致性,分三个等级:

级别说明实现方法
基本保证异常发生后,对象仍有效且无资源泄漏。依赖RAII(如智能指针、文件句柄)
强保证操作要么完全成功,要么状态回滚到操作前。事务语义或拷贝-交换惯用法
不抛保证函数保证不抛出异常。使用noexcept声明

 实现强保证的示例

void updateData() {auto backup = data_; // 备份原始数据try {modifyData();     // 可能抛出异常的操作} catch (...) {data_ = backup;   // 失败时恢复备份throw;}
}

​8. 异常处理与资源管理(RAII)​

通过 ​资源获取即初始化(RAII)​ 自动释放资源(如内存、文件句柄),避免异常导致泄漏:

#include <memory>
#include <fstream>void processFile(const std::string& filename) {std::ofstream file(filename); // 文件自动管理if (!file) {throw std::runtime_error("无法打开文件");}// 使用 file 对象,即使抛出异常也会自动关闭文件
}

​9. 综合示例
#include <iostream>
#include <stdexcept>class NetworkError : public std::runtime_error {
public:NetworkError() : std::runtime_error("网络连接失败") {}
};void connectToServer() {bool connectionFailed = true;if (connectionFailed) {throw NetworkError();}
}int main() {try {connectToServer();} catch (const NetworkError &e) {std::cerr << "错误: " << e.what() << std::endl;// 尝试重连或终止程序} catch (const std::exception &e) {std::cerr << "其他错误: " << e.what() << std::endl;}return 0;
}

10. 注意事项
  • 析构函数中的异常:析构函数默认应为 noexcept,若抛出异常可能导致程序终止。
  • 性能开销:异常处理比返回错误码略慢,但现代编译器优化后差异较小。
  • 适用场景:适合处理不可恢复的错误(如文件缺失、内存不足)。

11. 总结
关键点说明
分离错误处理逻辑使主流程代码更清晰
栈展开与 RAII确保资源自动释放,避免泄漏
自定义异常类提供更清晰的错误信息分类
noexcept 优化声明不抛异常的函数以提升性能
异常安全等级根据需求选择基本、强或无异常保证

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

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

相关文章

B站pwn教程笔记-5

复习和回顾 首先复习一下ELF文件在内存和磁盘中的不同。内存只关注读写这权限&#xff0c;会合并一些代码段。 动态链接库只在内存中单独装在一份 因为很多软件都要用动态链接库了&#xff0c;不可能一个个单独复制一份。但是在有的调试环境下会单独显示出来各一份。 ld.so是装…

Vue Router 的核心实现原理是什么?

文章目录 一、路由模式实现原理1. Hash 模式2. History 模式 二、响应式路由系统1. 路由对象响应化2. 路由映射解析 三、组件渲染机制1. RouterView 实现2. 路由匹配流程 四、导航守卫系统1. 守卫执行流程2. 守卫类型对比 五、核心源码结构六、设计亮点分析七、性能优化策略总结…

CherryStudio + 火山引擎DeepSeek R1 告别服务器繁忙

CherryStudio 火山引擎DeepSeek R1 告别服务器繁忙 一、下载CherryStudio并安装 CherryStudio是功能强大的多模型桌面客户端&#xff0c;支持Windows、macOS和Linux系统。集成了多种主流的大语言模型&#xff08;如OpenAI、DeepSeek、Gemini等&#xff09;以及本地模型运行功…

Hessian 矩阵是什么

Hessian 矩阵是什么 目录 Hessian 矩阵是什么Hessian 矩阵的性质及举例说明**1. 对称性****2. 正定性决定极值类型****特征值为 2(正),因此原点 ( 0 , 0 ) (0, 0) (0,0) 是极小值点。****3. 牛顿法中的应用****4. 特征值与曲率方向****5. 机器学习中的实际意义**一、定义与…

C#从入门到精通(1)

目录 第一章 C#与VS介绍 第二章 第一个C#程序 &#xff08;1&#xff09;C#程序基本组成 1.命名空间 2.类 3.Main方法 4.注释 5.语句 6.标识符及关键字 &#xff08;2&#xff09;程序编写规范 1.代码编写规则 2.程序命名方法 3.元素命名规范 第三章 变量 &…

【LINUX操作系统】 动静态库的链接原理

初识linux&#xff08;16&#xff09; 动静态库&#xff08;手搓动静态库&#xff01;&#xff09;-CSDN博客 完成了对动静态库使用的学习&#xff0c;现在浅显理解下动态库加载的原理。 1. 宏观认知 磁盘中的应用程序main和动态库libmystdio.so先加载到内存中 加载到内存后&am…

广东启动“跨境电商+产业带”系列活动 三年打造30个产业振兴样板

大湾区经济网湾区财经快讯&#xff0c;近日&#xff0c;2025年广东省“跨境电商&#xff0b;产业带”助力“百千万工程”系列活动在中山市古镇镇启动。作为外贸领域新质生产力的重要载体&#xff0c;跨境电商将通过赋能县域特色产业带转型升级&#xff0c;为城乡融合与乡村振兴…

穿透递归的本质:从无限梦境到可控魔法的蜕变之路

穿透递归的本质&#xff1a;从无限梦境到可控魔法的蜕变之路&#xff08;C实现&#xff09; 一、递归&#xff1a;程序员的盗梦空间 在计算机科学的宇宙中&#xff0c;递归是最接近魔法本质的编程范式。它像一面镜子中的镜子&#xff0c;引导我们通过自我相似性破解复杂问题。…

基于django+vue的购物商城系统

开发语言&#xff1a;Python框架&#xff1a;djangoPython版本&#xff1a;python3.8数据库&#xff1a;mysql 5.7数据库工具&#xff1a;Navicat11开发软件&#xff1a;PyCharm 系统展示 系统首页 热卖商品 优惠资讯 个人中心 后台登录 管理员功能界面 用户管理 商品分类管理…

WordPress靶场攻略

后台修改模板 修改404.php为一句话木马 访问404.php,验证有没有成功 http://47.122.51.245:8080/wp-content/themes/twentyfifteen/404.php?cmdphpinfo(); 上传主题 创建6.php,写入图中代码 自己随便下载一个主题包&#xff0c;将1.php和主题包压缩在一起&#xff0c;提交上…

JVM常用概念之对象对齐

问题 对象对齐有什么规范吗?对象对齐是8个字节吗? 基础知识 许多硬件实现要求对数据的访问是对齐的&#xff0c;即确保所有 N 字节宽度的访问都在 N 的整数倍的地址上完成。即使对于普通的数据访问没有特别要求&#xff0c;特殊操作&#xff08;特别是原子操作&#xff09…

K8S学习之基础三十七:prometheus监控node资源

Prometheus v2.2.1 ​ 编写yaml文件&#xff0c;包含创建ns、configmap、deployment、service # 创建monitoring空间 vi prometheus-ns.yaml apiVersion: v1 kind: Namespace metadata:name: monitor-sa# 创建SA并绑定权限 kubectl create serviceaccount monitor -n monito…

leetcode127.单词接龙

本题的思路就是将所有可转换的序列相连&#xff0c;构成图&#xff0c;然后选择起始词作为广度优先遍历的起点&#xff0c;那么就能找到转换的最小步骤数 而这里的两个单词是否相连不是真的把他们弄成一张图&#xff0c;而是采用暴力枚举&#xff0c;逐个尝试替换字母&#xf…

Tr0ll2靶机详解

一、主机发现 arp-scan -l靶机ip&#xff1a;192.168.55.164 二、端口扫描、漏洞扫描、目录枚举、指纹识别 2.1端口扫描 nmap --min-rate 10000 -p- 192.168.55.164发现21端口的ftp服务开启 以UDP协议进行扫描 使用参数-sU进行UDP扫描 nmap -sU --min-rate 10000 -p- 19…

Pycharm接入DeepSeek,提升自动化脚本的写作效率

一.效果展示&#xff1a; 二.实施步骤&#xff1a; 1.DeepSeek官网创建API key&#xff1a; 创建成功后&#xff0c;会生成一个API key&#xff1a; 2. PyCharm工具&#xff0c;打开文件->设置->插件&#xff0c;搜索“Continue”&#xff0c;点击安装 3.安装完成后&…

如何在 Node.js 中使用 .env 文件管理环境变量 ?

Node.js 应用程序通常依赖于环境变量来管理敏感信息或配置设置。.env 文件已经成为一种流行的本地管理这些变量的方法&#xff0c;而无需在代码存储库中公开它们。本文将探讨 .env 文件为什么重要&#xff0c;以及如何在 Node.js 应用程序中有效的使用它。 为什么使用 .env 文…

《视觉SLAM十四讲》ch13 设计SLAM系统 相机轨迹实现

前言 相信大家在slam学习中&#xff0c;一定会遇到slam系统的性能评估问题。虽然有EVO这样的开源评估工具&#xff0c;我们也需要自己了解系统生成的trajectory.txt的含义&#xff0c;方便我们更好的理解相机的运行跟踪过程。 项目配置如下&#xff1a; 数据解读&#xff1a; …

软考高级信息系统管理工程师通关100题(21-40)附记忆口诀

文章目录 21.常用存储模式的技术与应用对比22.物联网架构23.云计算服务提供的资源层次24.大数据25.区块链26.人工智能27.虚拟现实VR28.IT治理的内涵29.IT 治理活动30.IT治理本质31.IT审计目标32.IT审计方法33.治理系统设计34.数据管理能力成熟度评估模型35.项目管理原则36.管理…

Redisson 分布式锁原理

加锁原理 # 如果锁不存在 if (redis.call(exists, KEYS[1]) 0) then# hash结构,锁名称为key,线程唯一标识为itemKey&#xff0c;itemValue为一个计数器。支持相同客户端线程可重入,每次加锁计数器1.redis.call(hincrby, KEYS[1], ARGV[2], 1);# 设置过期时间redis.call(pexpi…