C++基础与深度解析 | 异常处理 | 枚举与联合 | 嵌套类与局部类 | 嵌套名字空间与匿名名字空间 | 位域与volatile关键字
文章目录
- 一、异常处理
- 二、枚举与联合
- 三、嵌套类与局部类
- 四、嵌套名字空间与匿名名字空间
- 五、位域与volatile关键字
一、异常处理
异常处理用于处理程序在调用过程中的非正常行为。
-
传统的处理方法:传返回值表示函数调用是否正常结束。
例如,返回
0
表示成功,非0
表示失败。这种方法的缺点是函数的返回值被错误处理逻辑占用,不能用于其他目的。这种方法有局限性
-
C++ 中的处理方法:通过关键字 try/catch/throw 引入异常处理机制
通过
try/catch/throw
引入了结构化的错误处理机制,使得错误处理逻辑与正常逻辑分离,提高了代码的可读性和可维护性。C++异常处理的关键组件:
-
throw关键字
throw
关键字用于抛出一个异常。它后面可以跟任意类型的表达式,该表达式的结果将被用作异常对象 -
try块
try
块包含了可能会抛出异常的代码。如果try
块中的代码抛出了异常,那么与之匹配的catch
块将被执行。 -
catch块
catch
块用于捕获并处理异常。可以有多个catch
块来捕获不同类型的异常。 -
异常类
C++标准库中定义了一些基本的异常类,如
std::exception
、std::runtime_error
、std::logic_error
等
-
异常触发时的系统行为—栈展开:
-
抛出异常后续的代码不会被执行
一旦抛出异常,
throw
语句之后的代码将不会被执行。控制流会立即转移到异常处理机制。 -
局部对象会按照构造相反的顺序自动销毁
在栈展开过程中,局部对象(包括由
new
分配的对象)会按照它们构造的相反顺序自动销毁。这是为了保证资源的正确释放,防止内存泄漏。 -
系统尝试匹配相应的 catch 代码段
-
如果找到匹配的
catch
块:- 执行
catch
块中的异常处理逻辑。 - 异常被“捕获”后,
catch
块之后的代码会继续执行。
- 执行
-
如果没有找到匹配的
catch
块:-
栈展开会继续进行,直到找到匹配的
catch
块或者退出当前函数。 -
如果当前函数中没有找到匹配的
块,栈展开会继续,直到:
-
找到一个匹配的
catch
块。 -
达到
main
函数。如果在
main
函数之前的所有函数中都没有找到匹配的catch
块,程序将退出main
函数。如果程序在退出main
函数之前没有捕获到异常,std::terminate
函数将被调用。这通常会导致程序立即终止。
-
-
-
异常对象:
- 系统会使用抛出的异常拷贝初始化一个临时对象,称为异常对象
- 异常对象会在栈展开过程中被保留,并最终传递给匹配的 catch 语句
try / catch语句块:
-
一个 try 语句块后面可以跟一到多个 catch 语句块(至少跟一个)
-
每个 catch 语句块用于匹配一种类型的异常对象
可以有多个
catch
块,每个用于处理不同类型的异常。 -
catch 语句块的匹配按照从上到下进行
catch
块按照它们在代码中出现的顺序(从上到下)进行匹配。一旦找到匹配的异常类型,就执行相应的catch
块,忽略后面的catch
块。 -
使用 catch(…) 匹配任意异常
catch(...)
是一个通用的异常捕获器,它可以捕获任何类型的异常,包括未被前面的catch
块捕获的异常。 -
在 catch 中调用 throw 继续抛出相同的异常
在
catch
块中,可以使用throw;
(不带参数)来重新抛出当前捕获的异常,这将导致继续搜索外层catch
块。
示例:
#include <iostream>struct Str{};
struct Base{};
struct Derive : Base{};void f1()
{int x;Str obj;//throw Derive{}; //打印Derive exception is called in f2 throw Str{}; //打印exception is called in f2
}void f2()
{int x2;Str obj2;try{f1();}catch(Derive& e){std::cout << "Derive exception is called in f2 " << "\n";}catch(Base& e){std::cout << "Base exception is called in f2 " << "\n";}catch(...){std::cout << "exception is called in f2" << "\n";throw; //重新抛出当前捕获的异常}std::cout << "other logic in f2.\n";
}void f3()
{try{f2(); }catch(Str& e){std::cout << "exception is called in f2" << "\n";}
}int main()
{f3();
}
在一个异常未处理完成时抛出新的异常会导致程序崩溃:
- 不要在析构函数或 operator delete 函数重载版本中抛出异常
- 通常来说, catch 所接收的异常类型为引用类型
异常与构造、析构函数:
-
使用 function-try-block保护初始化逻辑
在C++中,function-try-block允许你在函数的初始化列表和函数体中使用
try
和catch
。这在构造函数中特别有用,因为它可以保护对象的初始化代码。示例:
#include <iostream>struct Str {Str() { throw 100; } }class Resource { public:Resource() try : m_str(){}catch(int){std::cout << "Exception is catched at Resource::Resource" << std::endl;throw;}private:Str m_str; };int main() {try{Resource obj; }catch(int){std::cout << "Exception is catched at main" << std::endl;} }
运行结果:
Exception is catched at Resource::Resource Exception is catched at main
-
在构造函数中抛出异常:
-
已经构造的成员对象会被销毁
如果在构造函数中抛出异常,已经构造的成员对象将按照它们构造的相反顺序自动销毁。
-
类本身的析构函数不会被调用
如果异常是在对象的构造过程中抛出的,并且没有被捕获,那么类的析构函数不会被调用。这是因为对象被视为未完全构造,因此析构函数不适用。
-
局部对象的销毁
如果对象是局部的(即在栈上),异常抛出时,局部对象会自动销毁,但不会调用其析构函数。
-
动态分配对象的销毁
如果对象是动态分配的(即使用
new
),在构造函数中抛出异常且未被捕获时,需要手动释放分配的内存,因为析构函数不会被调用。
-
描述函数是否会抛出异常:
-
如果函数不会抛出异常,则应表明,从而为系统提供更多的优化空间
- C++ 98 的方式:
- throw() :表明不会抛出异常
- throw(int, char):表明可能抛出异常,显式给定了要抛出异常的类型
- C++11 后的改进:
- noexcept :表明不会抛出异常
- noexcept(false):表明可能抛出异常,不需要显式给定会抛出哪种类型的异常
- C++ 98 的方式:
-
noexcept
- 限定符:接收 false / true 表示是否会抛出异常
- 操作符:接收一个表达式,根据表达式是否可能抛出异常返回 false/true
- 在声明了 noexcept 的函数中抛出异常会导致 terminate 被调用,程序终止
- 不作为函数重载依据,但函数指针、虚拟函数重写时要保持形式兼容
示例:
#include <iostream>void fun2()
{}void fun() noexcept(noexcept(fun2()))
{fun2();
}int main()
{std::cout << noexcept(fun()) << std::endl;
}
二、枚举与联合
三、嵌套类与局部类
四、嵌套名字空间与匿名名字空间
五、位域与volatile关键字
相关文章:
C++基础与深度解析 | 异常处理 | 枚举与联合 | 嵌套类与局部类 | 嵌套名字空间与匿名名字空间 | 位域与volatile关键字
文章目录 一、异常处理二、枚举与联合三、嵌套类与局部类四、嵌套名字空间与匿名名字空间五、位域与volatile关键字 一、异常处理 异常处理用于处理程序在调用过程中的非正常行为。 传统的处理方法:传返回值表示函数调用是否正常结束。 例如,返回 0 表示…...

番外篇 | 利用华为2023最新Gold-YOLO中的Gatherand-Distribute对特征融合模块进行改进
前言:Hello大家好,我是小哥谈。论文提出一种改进的信息融合机制Gather-and-Distribute (GD) ,通过全局融合多层特征并将全局信息注入高层,以提高YOLO系列模型的信息融合能力和检测性能。通过引入MAE-style预训练方法,进一步提高模型的准确性。🌈 目录 🚀1.论文解…...
python记录之字符串
在Python中,字符串是一种非常常见且重要的数据类型,用于存储文本信息。下面,我们将对Python字符串进行深入的讲解,包括其基本操作、常见方法、格式化以及高级特性。 1. 字符串的创建 在Python中,字符串可以通过单引号…...

Elasticsearch 认证模拟题 - 15
一、题目 原索引 task1 的字段 title 字段包含单词 The,查询 the 可以查出 1200 篇文档。重建 task1 索引为 task1_new,重建后的索引, title 字段查询 the 单词,不能匹配到任何文档。 PUT task1 {"mappings": {"…...
g++ 预处理 编译 汇编 链接 命令
g 预处理 编译 汇编 链接 命令 在命令行中使用 g 预处理、编译、汇编和链接源代码文件通常遵循以下步骤: 预处理(Preprocessing):将源代码文件转换为经过预处理器处理的中间文件。 g -E source.cpp -o source.i 编译ÿ…...
计算机视觉中的low-level与 high-level任务
文章目录 low-level任务high-level任务区别联系others参考在计算机视觉领域中,low-level任务和high-level任务是两个重要的概念,他们分别涉及图像处理和分析的不同的层次。 low-level任务 low-level任务主要关注的是图像的底层特征,如颜色、纹理、边缘、形状等。通常涉及对…...

安徽京准NTP时钟系统:GPS北斗卫星授时下的生活重塑
安徽京准NTP时钟系统:GPS北斗卫星授时下的生活重塑 安徽京准NTP时钟系统:GPS北斗卫星授时下的生活重塑 时间的流逝自古以来时钟都是人类生活与活动的基础。然而,随着科技的进步,我们对时间管理和测量的方法已经发生了翻天覆地的变…...
图论第8天
685.冗余连接II 这题需要考虑两种情况: 1.两个输入 2.没有两个输入就是有成环 class Solution { public:static const int N 1005;int father[N];int n;void init(){for (int i 0; i < n; i){father[i] i;}}int find(int x){return x father[x] ? x : f…...
Python怎么配置环境变量:深度探索与实战指南
Python怎么配置环境变量:深度探索与实战指南 在Python编程的世界中,环境变量的配置是一个至关重要的步骤。它不仅影响着Python解释器的运行,还关系到各种第三方库和工具的使用。本文将带你深度探索如何配置Python的环境变量,并为…...

计网期末复习指南(六):应用层(DNS、FTP、URL、HTTP、SMTP、POP3)
前言:本系列文章旨在通过TCP/IP协议簇自下而上的梳理大致的知识点,从计算机网络体系结构出发到应用层,每一个协议层通过一篇文章进行总结,本系列正在持续更新中... 计网期末复习指南(一):计算…...

HTML做成一个炫酷跳动爱心的页面
大家好,今天制作制作一个炫酷跳动爱心的页面! 先看具体效果: 要创建一个炫酷跳动爱心的HTML页面,你可以使用HTML、CSS和JavaScript的组合。以下是一个简单的示例,它使用CSS动画和JavaScript来实现跳动效果。 首先&…...

React + SpringBoot实现图片预览和视频在线播放,其中视频实现切片保存和分段播放
图片预览和视频在线播放 需求描述 实现播放视频的需求时,往往是前端直接加载一个mp4文件,这样做法在遇到视频文件较大时,容易造成卡顿,不能及时加载出来。我们可以将视频进行切片,然后分段加载。播放一点加载一点&am…...

Suse Linux ssh配置免密后仍需要输入密码
【问题描述】 Suse Linux已经配置了ssh免密,但无法ssh到目标服务器。 对自身的ssh登陆也需要输入密码。 系统–Suse 15 SP5 【重现步骤】 1.使用ssh-keygen -t rsa生产key文件 2.使用ssh-copy-id拷贝public key到目标机器(或者自身) 3.配置成功后ssh 目标时仍需要输…...

apifox 生成签名
目录 前言准备编写签名脚本签名说明捋清思路编码获取签名所需的参数生成签名将签名放到合适的位置完整代码 在apifox中配置脚本新增公共脚本引用公共脚本添加环境变量 参考 前言 略 准备 查看apifox提供的最佳实践文章:接口签名如何处理 编写签名脚本 签名说明…...
介绍建造者模式
建造者模式 将一个复杂对象的创建与它的表示分离,使得同样的构建过程可以创建不同的表示 四种角色 Product 产品角色 指的是一个具体的产品对象Builder 抽象建造者 创建一个产品对象的各个部件的接口/抽象类ConcreteBuilder 具体建造者 实现或继承抽象建造者接口…...

【全部更新完毕】2024全国大学生数据统计与分析竞赛B题思路代码文章教学数学建模-电信银行卡诈骗的数据分析
电信银行卡诈骗的数据分析 摘要 电信银行卡诈骗是当前社会中严重的犯罪问题,分析电信银行卡交易数据,找出高风险交易特征,建立预测模型,将有助于公安部门和金融机构更好地防范诈骗行为,保障用户的财产安全。 针对问…...

【应用浅谈】Odoo的库存计价与产品成本(三)
序言:时间是我们最宝贵的财富,珍惜手上的每个时分 Odoo的库存(Stock)模块拥有众多功能,其中库存计价是一项非常重要的功能,原生的成本方法分三种:【标准成本】,【平均成本】,【先进先出】&#…...

数据结构之ArrayList与顺序表(下)
找往期文章包括但不限于本期文章中不懂的知识点: 个人主页:我要学编程(ಥ_ಥ)-CSDN博客 所属专栏:数据结构(Java版) 目录 ArrayList的具体使用 118. 杨辉三角 扑克洗牌算法 接上篇:数据结构之ArrayLis…...
openi启智社区 aarch64 npu环境安装飞桨paddlepaddle和PaddleNLP(失败)
以前在启智社区都是编译安装飞桨,这回看到飞桨提供了npu安装包,兴冲冲的以为安装很简单。 之所以安装飞桨,是因为想在启智社区的启智大脑调试环境使用最新的PaddleNLP,结果报错:No module named paddle.nn.layer.laye…...

【漏洞复现】多客圈子论坛系统 httpGet 任意文件读取漏洞
0x01 产品简介 多客圈子论坛系统是一种面向特定人群或特定话题的社交网络,它提供了用户之间交流、分享、讨论的平台。在这个系统中,用户可以创建、加入不同的圈子,圈子可以是基于兴趣、地域、职业等不同主题的。用户可以在圈子中发帖、评论、…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...

《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)
CSI-2 协议详细解析 (一) 1. CSI-2层定义(CSI-2 Layer Definitions) 分层结构 :CSI-2协议分为6层: 物理层(PHY Layer) : 定义电气特性、时钟机制和传输介质(导线&#…...
土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等
🔍 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术,可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势,还能有效评价重大生态工程…...

视觉slam十四讲实践部分记录——ch2、ch3
ch2 一、使用g++编译.cpp为可执行文件并运行(P30) g++ helloSLAM.cpp ./a.out运行 二、使用cmake编译 mkdir build cd build cmake .. makeCMakeCache.txt 文件仍然指向旧的目录。这表明在源代码目录中可能还存在旧的 CMakeCache.txt 文件,或者在构建过程中仍然引用了旧的路…...

三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...
Docker拉取MySQL后数据库连接失败的解决方案
在使用Docker部署MySQL时,拉取并启动容器后,有时可能会遇到数据库连接失败的问题。这种问题可能由多种原因导致,包括配置错误、网络设置问题、权限问题等。本文将分析可能的原因,并提供解决方案。 一、确认MySQL容器的运行状态 …...

Mysql故障排插与环境优化
前置知识点 最上层是一些客户端和连接服务,包含本 sock 通信和大多数jiyukehuduan/服务端工具实现的TCP/IP通信。主要完成一些简介处理、授权认证、及相关的安全方案等。在该层上引入了线程池的概念,为通过安全认证接入的客户端提供线程。同样在该层上可…...
Django RBAC项目后端实战 - 03 DRF权限控制实现
项目背景 在上一篇文章中,我们完成了JWT认证系统的集成。本篇文章将实现基于Redis的RBAC权限控制系统,为系统提供细粒度的权限控制。 开发目标 实现基于Redis的权限缓存机制开发DRF权限控制类实现权限管理API配置权限白名单 前置配置 在开始开发权限…...
Easy Excel
Easy Excel 一、依赖引入二、基本使用1. 定义实体类(导入/导出共用)2. 写 Excel3. 读 Excel 三、常用注解说明(完整列表)四、进阶:自定义转换器(Converter) 其它自定义转换器没生效 Easy Excel在…...
Qt学习及使用_第1部分_认识Qt---Qt开发基本流程
前言 学以致用,通过QT框架的学习,一边实践,一边探索编程的方方面面. 参考书:<Qt 6 C开发指南>(以下称"本书") 标识说明:概念用粗体倾斜.重点内容用(加粗黑体)---重点内容(红字)---重点内容(加粗红字), 本书原话内容用深蓝色标识,比较重要的内容用加粗倾…...