c++从入门到精通(五)--异常处理,命名空间,多继承与虚继承

news2025/5/19 20:14:21

异常处理

栈展开过程: 栈展开过程沿着嵌套函数的调用链不断查找,直到找到了与异常匹配的catch子句为止;也可能一直没找到匹配的catch,则退出主函数后查找过程终止。栈展开过程中的对象被自动销毁。

在栈展开的过程中,会自动运行类类型的局部对象的析构函数,这些异构函数不应该抛出异常。一旦在栈展开的过程中析构函数抛出了异常,且析构函数自身没能捕获到该异常,则程序将被终止。

抛出指针异常,要求在任何对应的处理代码存在的地方,指针所指的对象都必须存在。

捕获异常:

  • 声明的类型决定了处理代码所能捕获的异常类型。这个类型必须是完全类型,它可以是左值引用,但不能是右值引用。

  • 最后一点需要注意的是,异常声明的静态类型将决定catch语句所能执行的操作。如果catch的参数是基类类型,则catch无法使用派生类特有的任何成员。

  • 通常情况下,如果catch接受的异常与某个继承体系有关,则最好将该catch的参数定义成引用类型。

  • 因为catch语句是按照其出现的顺序逐一进行匹配的,所以当程序使用具有继承关系的多个异常时必须对catch语句的顺序进行组织和管理,使得派生类异常的处理代码出现在基类异常的处理代码之前。

  • 与实参和形参的匹配规则相比,异常和catch异常声明的匹配规则受到更多限制。此时,绝大多数类型转换都不被允许,除了一些极细小的差别之外,要求异常的类型和catch声明的类型是精确匹配的:

    【非常量向常量的转换,派生类向基类的转换,数组被转换为指针,函数被转换为指向该函数的指针】

    除此之外,包括标准算术类型转换和类类型转换在内,其他所有转换规则都不能在匹配catch的过程中使用。

重新抛出

一条catch语句通过重新抛出(rethrowing)的操作将异常传递给另外一个catch语句。这里的重新抛出仍然是一条throw语句,只不过不包含任何表达式:throw。一个重新抛出语句并不指定新的表达式,而是将当前的异常对象沿着调用链向上传递。

如果catch语句改变了参数内容,重新抛出异常后,只有当catch异常声明是引用类型时,我们对参数的修改才会被保留并继续传播。

捕获所有异常

我们使用省略号作为异常声明catch(...),通常于重新抛出语句一起使用,其中catch执行当前局部能能完成的工作,随后重新抛出异常。

如果catch(…)与其他几个catch语句一起出现,则catch(…)必须在最后的位置。出现在捕获所有异常语句后面的catch语句将永远不会被匹配。

函数try语句块于构造函数:

要想处理构造函数初始值抛出的异常,我们必须将构造函数写成函数try语句块。语句块。与这个try关联的catch既能处理构造函数体抛出的异常,也能处理成员初始化列表抛出的异常。

还有一种情况值得读者注意,在初始化构造函数的参数时也可能发生异常,这样的异常不属于函数try语句块的一部分。函数try语句块只能处理构造函数开始执行后发生的异常。和其他函数调用一样,如果在参数初始化的过程中发生了异常,则该异常属于调用表达式的一部分,并将在调用者所在的上下文中处理。

template <typenameT>
Blob<T>::Blob(std::initializer_list<T> il) try:data(std::make_shared<std::vector<T>>(il)){}catch(const std::bat_alloc &e){handle_out_of_memory(e);}

noexpect说明符:

对于一个函数来说,noexcept说明要么出现在该函数的所有声明语句和定义语句中,要么一次也不出现。该说明应该在函数的尾置返回类型之前。==我们也可以在函数指针的声明和定义中指定noexcept。在typedef或类型别名中则不能出现noexcept。==在成员函数中,noexcept说明符需要跟在const及引用限定符之后,而在final、override或虚函数的=0之前。

一旦一个noexcept函数抛出了异常,程序就会调用terminate以确保遵守不在运行时抛出异常的承诺

noexcept表达式:

返回一个bool类型的右值常量表达式,用于表示给定的表达式是否会抛出异常,和sizeof类似,noexcept不会求其运算对象的值。noexcept(recoup(i))如果recoup不抛出异常则为true。

在这里插入图片描述

指针,函数,拷贝控制与异常说明

如果我们为某个指针做了不抛出异常的声明,则该指针将只能指向不抛出异常的函数。

如果一个虚函数承诺了它不会抛出异常,则后续派生出来的虚函数也必须做出同样的承诺

当编译器合成拷贝控制成员时,同时也生成一个异常说明。如果对所有成员和基类的所有操作都承诺了不会抛出异常,则合成的成员是noexcept的。

异常类的层次:

在这里插入图片描述

命名空间

​ 一个命名空间的定义包含两部分:首先是关键字namespace,随后是命名空间的名字。在命名空间名字后面是一系列由花括号括起来的声明和定义。只要能出现在全局作用域中的声明就能置于命名空间内,主要包括:类、变量(及其初始化操作)、函数(及其定义)、模板和其他命名空间命名空间作用域后面无须分号。类的作用域后面又分号

​ 命名空间的定义可以是不连续的,我们可以新定义一个命名空间,也可以为已存在的命名空间添加一些新成员。因此我们可以把类的声明写在头文件中的命名空间中,把类的定义写在源文件的命名空间中。

​ 我们可以为一个项目定义一个命名空间,在该命名空间中定义不同的类,每个类都单独的头文件和定义源文件。使用的时候需要包含指定的头文件,并且应用该项目的命名空间。

模板特例化必须定义在原始模板所属的命名空间中。我们必须将模板特例化声明为std的成员(特例化std命名空间内的模板)

全局命名空间:

::name表示全局命名空间中的名字。

内联命名空间

​ 内联命名空间中的名字可以直接被外层命名空间访问。inline namesapce FifthEd{}关键字inline必须出现在命名空间第一次定义的地方,后续再打开命名空间的时候可以写inline,也可以不写。内联命名空间可以作为版本控制。

在这里插入图片描述

未命名的命名空间

​ 未命名命名空间在关键字namespace后直接跟花括号。未命名的命名空间中定义的变量拥有静态生命周期:它们在第一次使用前创建,并且直到程序结束才销毁。

​ 如果一个头文件定义了未命名的命名空间,则该命名空间中定义的名字将在每个包含了该头文件的文件中对应不同实体。

​ 和其他命名空间不同,未命名的命名空间仅在特定的文件内部有效,其作用范围不会横跨多个不同的文件。

​ 未命名的命名空间中定义的名字的作用域与该命名空间所在的作用域相同。如果未命名的命名空间定义在文件的最外层作用域中,则该命名空间中的名字一定要与全局作用域中的名字有所区别。

​ 未命名的命名空间可以嵌套使用,我们可以用外围命名空间的名字访问未命名的命名空间中的名字

定义名字别名

namespace asodfjdlkasjflksadjflk primer使用primer作为别名

namespace Qlib=asdlkfjlkasdjflk::asdfasd

using声明和using指示

​ using声明using std::vector。using声明会扩展候选函数集的规模。using声明的函数如果和当前作用域中的函数完全一致则using声明报错

​ using指示using namespace std. 如果存在多个using指示。则来自每个命名空间的名字都会成为候选函数集的一部分。using指示可以引入和当前作用域中完全重名的函数,我们只需要用作用域运算符就可以访问命名空间中的函数

​ using声明和using指示在作用域上的区别直接决定了它们工作方式的不同。对于using声明来说,我们只是简单地令名字在局部作用域内有效。相反,using指示是令整个命名空间的所有内容变得有效。它具有将命名空间成员提升到包含命名空间本身和using指示的最近作用域的能力。

在这里插入图片描述

​ 尽量避免使用using指示,在命名空间的定义源文件中可以使用using指示来简化编码。

函数实参传递和命名空间:

​ 当我们给函数传递一个类类型的对象时,除了在常规的作用域查找外还会查找实参类所属的命名空间。这一例外对于传递类的引用或指针的调用同样有效。查找规则的这个例外允许概念上作为类接口一部分的非成员函数无须单独的using声明就能被程序使用。

​ 对于标准库中的move和forward函数,他们几乎可以接收任何一个形参,因此我们编写的move和forward函数几乎一定会和标准库的相关函数冲突。因此我们通常书写std::move(std::forward)。

友元与命名空间

一个未声明的类或函数如果第一次出现在友元声明中,则我们认为它是最近的外层命名空间的成员。f2和f被隐式声明为A的成员。

在这里插入图片描述

​ 上述代码中,我们声明了一个C类的一个实例,调用f的时候会在C类所在的命名空间中查找(因为f接收一个类类型的实参),此时会找到f的隐士声明(在C类之前)。调用f2它不接受类类型参数,也因此不会查找C类所在的命名空间,也就找不到f2的声明语句。

多继承与虚继承

**继承的构造函数与多继承:**如果一个类从他的多个基类中继承了相同的构造函数,则这个类必须为该构造函数定义它自己的版本。下面代码中Base1和Base2都有一个接受const string &s的构造函数。

在这里插入图片描述

拷贝控制:多重继承的派生类如果定义了自己的拷贝/赋值构造函数和赋值运算符,则必须在完整的对象上执行拷贝、移动或赋值操作。只有当派生类使用的是合成版本的拷贝、移动或赋值成员时,才会自动对其基类部分执行这些操作。

**派生类向基类的转换:**在多继承关系中,编译器认为派生类向任何一个基类的转换都是一样好。因此函数调用过程中可能存在二义性错误。

对象、指针和引用的静态类型决定了我们能使用那些成员。

**多继承下的类作用域:**在多重继承的情况下,相同的查找过程在所有直接基类中同时进行。如果名字在多个基类中都被找到,则对该名字的使用将具有二义性。 对于一个派生类来说,从它的几个基类中分别继承名字相同的成员是完全合法的,只不过在使用这个名字时必须明确指出它的版本。

**虚继承:**避免在继承体系中,派生类里有多个基类(可以是顶层基类,也可以是中间基类)对象。

在默认情况下,派生类中含有继承链上每个类对应的子部分。如果某个类在派生过程中出现了多次,则派生类中将包含该类的多个子对象。

虚继承令某个类做出声明,承诺愿意共享它的基类。其中共享的基类对象称为虚基类。在这种机制下,无论虚基类在继承体系中出现了多少次,在派生类中都包含唯一一个共享的虚基类子对象。

在实际的编程过程中,位于中间层次的基类将其继承声明为虚继承一般不会带来什么问题。

无论基类是不是虚基类,派生类对象都能被可访问基类的指针或引用操作。

class Raccoon : public virtual ZooAnimal { }

**构造函数与虚继承:**派生类先

​ 含有虚基类的对象的构造顺序与一般的顺序稍有区别:首先使用提供给最低层派生类的构造函数的初始值初始化该对象的虚基类子部分,接下来按照直接基类在派生列表中出现的次序依次对其进行初始化。

如果派生类没有显示初始化虚基类,则虚基类的默认构造函数将被调用,如果虚基类没有默认构造函数,则代码将会发生错误。

编译器按照直接基类的声明顺序对其依次进行检查,以确定其中是否含有虚基类。如果有,则先构造虚基类,然后按照声明的顺序逐一构造其他非虚基类。

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

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

相关文章

gcc/g++常用参数

1.介绍 gcc用于编译c语言&#xff0c;g用于编译c 源代码生成可执行文件过程&#xff0c;预处理-编译-汇编-链接。https://zhuanlan.zhihu.com/p/476697014 2.常用参数说明 2.1编译过程控制 参数作用-oOutput&#xff0c;指定输出名字-cCompile&#xff0c;编译源文件生成对…

nginx配置之负载均衡

版权声明&#xff1a;原创作品&#xff0c;请勿转载&#xff01; 1.实验环境准备 准备3台linux服务器&#xff08;ubuntu和centos均可&#xff0c;本文使用centos7.9&#xff09;&#xff0c;两台web和一台负载均衡服务器&#xff0c;均安装nginx服务 主机名IP软件lb0110.0.0…

去年开发一款鸿蒙Next Os的window工具箱

持拖载多个鸿蒙应用 批量签名安装 运行 http://dl.lozn.top/lozn/HarmonySignAndFileManagerTool_2024-11-26.zip 同类型安卓工具箱以及其他软件下载地址汇总 http://dl.lozn.top/lozn/ 怎么个玩法呢&#xff0c;比如要启动某app, 拖载识别到包名 点启动他能主动读取包名 然后…

uniapp|实现获取手机摄像头权限,调用相机拍照实现人脸识别相似度对比,拍照保存至相册,多端兼容(APP/微信小程序)

基于uniapp以及微信小程序实现移动端人脸识别相似度对比,实现摄像头、相册权限获取、相机模块交互、第三方识别集成等功能,附完整代码。 目录 核心功能实现流程摄像头与相册权限申请权限拒绝后的引导策略摄像头调用拍照事件处理人脸识别集成图片预处理(Base64编码/压缩)调用…

JavaScript【7】BOM模型

1.概述&#xff1a; BOM&#xff08;Browser Object Model&#xff0c;浏览器对象模型&#xff09;是 JavaScript 中的一个重要概念&#xff0c;它提供了一系列对象来访问和操作浏览器的功能和信息。与 DOM&#xff08;Document Object Model&#xff09;主要关注文档结构不同&…

[强化学习的数学原理—赵世钰老师]学习笔记02-贝尔曼方程

本人为强化学习小白&#xff0c;为了在后续科研的过程中能够较好的结合强化学习来做相关研究&#xff0c;特意买了西湖大学赵世钰老师撰写的《强化学习数学原理》中文版这本书&#xff0c;并结合赵老师的讲解视频来学习和更深刻的理解强化学习相关概念&#xff0c;知识和算法技…

深入理解构造函数,析构函数

目录 1.引言 2.构造函数 1.概念 2.特性 3.析构函数 1.概念 2.特性 1.引言 如果一个类中什么都没有&#xff0c;叫作空类. class A {}; 那么我们这个类中真的是什么都没有吗?其实不是,如果我们类当中上面都不写.编译器会生成6个默认的成员函数。 默认成员函数:用户没有显…

Day 16

目录 1.JZ79 判断是不是平衡二叉树1.1 解析1.2 代码 2.DP10 最大子矩阵2.1 解析2.2 代码 1.JZ79 判断是不是平衡二叉树 JZ79 判断是不是平衡二叉树 dfs 1.1 解析 1.2 代码 /*** struct TreeNode {* int val;* struct TreeNode *left;* struct TreeNode *right;* TreeNode(in…

摄影构图小节

1、三分构图法 三分构图法即将画面横竖各分为三份&#xff0c;即九宫格形式。 将画面用两条竖线和两条横线分割&#xff0c;就如同是书写中文的【井】字。这样就可以得到4个交叉点&#xff0c;然后再将需要表现的重点放置在4个交叉点中的一个附近即可。 拍摄自然风光时&#xf…

DAY 28 类的定义和方法

知识点回顾&#xff1a; 类的定义pass占位语句类的初始化方法类的普通方法类的继承&#xff1a;属性的继承、方法的继承 比如def、class这些定义的关键词后&#xff0c;必须有一个有占据缩进位置的代码块。 还有下面这些依赖缩进的语句&#xff0c;都可以用pass语句来占位 x 1…

RAG数据处理:PDF/HTML

RAG而言用户输入的数据通常是各种各样文档&#xff0c;本文主要采用langchain实现PDF/HTML文档的处理方法 PDF文档解析 PDF文档很常见格式&#xff0c;但内部结构常常较复杂&#xff1a; 复杂的版式布局多样的元素&#xff08;段落、表格、公式、图片等&#xff09;文本流无…

机器学习 day04

文章目录 前言一、线性回归的基本概念二、损失函数三、最小二乘法 前言 通过今天的学习&#xff0c;我掌握了机器学习中的线性回归的相关基本概念&#xff0c;包括损失函数的概念&#xff0c;最小二乘法的理论与算法实现。 一、线性回归的基本概念 要理解什么是线性回归&…

蓝牙耳机什么牌子好?倍思值得冲不?

最近总被问“蓝牙耳机什么牌子好”&#xff0c;作为踩过无数坑的资深耳机党&#xff0c;必须安利刚入手的倍思M2s Pro主动降噪蓝牙耳机&#xff01;降噪、音质、颜值全都在线&#xff0c;性价比直接拉满。 -52dB降噪&#xff0c;通勤摸鱼神器 第一次开降噪就被惊到&#xff01…

机器学习-人与机器生数据的区分模型测试-数据处理 - 续

这里继续 机器学习-人与机器生数据的区分模型测试-数据处理1的内容 查看数据 中1的情况 #查看数据1的分布情况 one_ratio_list [] for col in data.columns:if col city or col target or col city2: # 跳过第一列continueelse:one_ratio data[col].mean() # 计算1值占…

ESP系列单片机选择指南:结合实际场景的最优选择方案

前言 在物联网(IoT)快速发展的今天&#xff0c;ESP系列单片机凭借其优异的无线连接能力和丰富的功能特性&#xff0c;已成为智能家居、智慧农业、工业自动化等领域的首选方案。本文将深入分析各款ESP芯片的特点&#xff0c;结合典型应用场景&#xff0c;帮助开发者做出最优选择…

特斯拉虚拟电厂:能源互联网时代的分布式革命

在双碳目标与能源转型的双重驱动下&#xff0c;特斯拉虚拟电厂&#xff08;Virtual Power Plant, VPP&#xff09;通过数字孪生技术与能源系统的深度融合&#xff0c;重构了传统电力系统的运行范式。本文从系统架构、工程实践、技术挑战三个维度&#xff0c;深度解析这一颠覆性…

【Linux笔记】nfs网络文件系统与autofs(nfsdata、autofs、autofs.conf、auto.master)

一、nfs概念 NFS&#xff08;Network File System&#xff0c;网络文件系统&#xff09; 是一种由 Sun Microsystems 于1984年开发的分布式文件系统协议&#xff0c;允许用户通过网络访问远程计算机上的文件&#xff0c;就像访问本地文件一样。它广泛应用于 Unix/Linux 系统&a…

博客打卡-求解流水线调度

题目如下&#xff1a; 有n个作业&#xff08;编号为1&#xff5e;n&#xff09;要在由两台机器M1和M2组成的流水线上完成加工。每个作业加工的顺序都是先在M1上加工&#xff0c;然后在M2上加工。M1和M2加工作业i所需的时间分别为ai和bi&#xff08;1≤i≤n&#xff09;。 流水…

数据库blog1_信息(数据)的处理与效率提升

&#x1f33f;信息的处理 &#x1f342;实际中离不开信息处理 ● 解决问题的建模 任何对问题的处理都可以看作数据的输入、处理、输出。 eg.一个项目中&#xff0c;用户点击信息由前端接收传递到后端处理后返回结果eg.面对一个问题&#xff0c;我们在搜集信息后做出处理与分析…

布隆过滤器介绍及其在大数据场景的应用

目录 布隆过滤器&#xff08;Bloom Filter&#xff09;介绍一、布隆过滤器的基本原理插入元素过程&#xff1a;查询元素过程&#xff1a; 二、布隆过滤器的特点三、误判率计算四、举例说明五、总结 Python版的简单布隆过滤器实现示例一、简单布隆过滤器Python示例二、布隆过滤器…