`
evasiu
  • 浏览: 170457 次
  • 性别: Icon_minigender_2
  • 来自: 广州
博客专栏
Fa47b089-e026-399c-b770-017349f619d5
TCP/IP详解卷一>阅读...
浏览量:12596
社区版块
存档分类
最新评论

C++ premier -- 异常、命名空间以及多重继承

 
阅读更多

放了两个星期的暑假,回来久久不能进入状态。也或许是这一章对编程经验的要求远高于我目前的水平,很难静下心来看下去。总结也可能会因此做得很不到位。Anyway,有些东西就先记在这里吧。

第17章起就是高级主题了,主要涉及大型程序中使用的工具,包括异常的使用、命名空间以及多重继承。

 

1.异常
通过异常我们可以将问题的检测和问题的解决分享,这样程序的问题检测部分可以不必了解如何处理问题。


1.1抛出异常对象throw
在C++的异常处理中,需要由问题检测部分抛出一个对象给处理代码,通过这个对象的类型和内容,两个部分能够就出现了什么错误进行通信,这就是异常对象。异常以类似于将实参传给函数的方式抛出和捕获。
执行throw的时候,不会执行跟在throw后面的语句,而是将控制从throw转移到匹配的catch,该catch可以是同一函数中局部的catch,也可以在直接或间接调用发生异常的函数的另一个函数中。控制从一个地方传到另一个地方,这有两个重要含义:
(1)沿着调用链的函数提早退出。
(2)一般而言,在处理异常的时候,抛出异常的块中的局部存储不存在了。
异常对象由编译器管理,而且保证驻留在可以被激活的任意catch都可以访问的空间。
当抛出一个表达式的时候,被抛出对象的静态编译时类型将决定异常对象的类型。也就是说,假如抛出的对象是基类的解引用,而该指针实际指向继承类,该对象会被分割,成为事实上的基类对象。
抛出异常的时候,交暂停当前函数的执行,开始查找匹配的catch子句。首先检查throw本身是否在try块内部,如果是,检查与该try相关的catch子句,看是否其中之一与被抛出对象相匹配。如果找到匹配的catch,就处理异常;如果找不到,就退出当前函数(释放当前函数的内存并撤销局部对象),并且继续在调用函数中查找。如果对抛出异常的函数的请用是在try块中,则检查与该try相关的catch子句。如果找到匹配的catch,就处理异常;如果找不到匹配的catch,调用函数也退出,并且继续在调用这个函数的函数中查找,这个过程称之为“栈展开”。
如果一个块直接分配资源,而且在释放资源之前发生异常,在栈展开期间将不会释放该资源。

 

1.2捕获异常catch
catch子句中的异常说明符看起来像只包含一个形参的形参表,异常说明符是在其后跟一个可选形参名的类型名。类型必须是完全类型,不能向前声明。
在查找匹配的catch期间,找到的catch不必是与异常最匹配的那个catch,相反,将选中第一个可以处理该异常的catch。异常与catch说明符匹配的规则只允许以下转换:
(1)允许从非const到const的转换。也就是说,非const对象的throw可以与指定接受const引用的catch匹配。
(2)允许从派生类类型到基类类型的转换
(3)将数组转换为指向数组类型的指针,将函数转换为指向函数类型的适当指针。
通常,如果catch子句处理因继承而相关的类型的异常,它就应该将自己的形参定义为引用。(其规则与一般函数一样)
如果单个catch不能完全处理一个异常,在进行了一些校正行动之后,catch可能确定该异常必须由函数调用链中更上层的函数来处理,catch可以通过重新抛出将异常传递给调用链中更上层的函数。

 

1.3 函数测试块与构造函数
构造函数函数体内部的catch子句不能处理在处理构造函数初始化式时可能发生的异常。构造函数要处理来自构造函数初始化式的异常,唯一的方法是将构造函数编写为函数测试块。

template<class T> Handle<T>::Handle(T *p)
	try: ptr(p), use(new size_t(1)){
		//empty function body
	}catch( const std::bad_alloc& e ){
		handle_out_of_memory(e);
	}

1.4用类管理资源分配
异常安全:异常安全意味着,即使发生异常,程序也有正确操作。在这种情况下,“安全”来自于保证“如果发生异常,被分配的任何资源都能适当地释放”。
例如下面的函数:

void exercise(int* b, int* e){
	vector<int> v(b,e);
	int *p=new int[v.size()];
	ifstream in("ints");
	//exception occurs
	//...
	delete p;	//p will not be released
}

用类管理资源分配的例子如下:

class Resource{
	public:
		Resource( size_t t ):r(new int[t]){}
		~Resource(){
			if(r)
				delete r;
			}
	private:
		int* r;
	};

void exercise( int *b, int *e ){
	vector<int> v(b,e);
	Resource res(v.size());
	ifstream in("ints");
	//now if exceptions occur here
	//destructor of res will be invoked 
	//to release the pointer in res
	...
}

标准库中提供的auto_ptr是“资源分配即初始化”技术的例子。

 

2. 命名空间
命名空间的作用是限定作用域。在命名空间外部使用命名空间的成员时,必须使用限定名namespace_name::member_name。命名空间可以是不连续的,一个命名空间的分离部分可以分散在多个文件中,由此可以用分离的接口文件和实现文件构成命名空间。
未命名的命名空间在定义时没有给定名字。未命名的命名空间以关键字namespace开头,后面直接将花括号。它可以在给定文件中不连续,但是不能跨越文件,每个文件有自己的未命名空间。
未命名的命名空间用于声明局部于文件的实体。在未命名的命名上定义的变量在程序开始时创建,在程序结束之前一直存在。(它将取代C中的static声明)
除了使用限定名namespace_name::member_name用引用命名空间的成员外,还可以使用using声明、using指示及命名空间别名的方法。
using声明一次只引入一个命名空间成员,如:

using std::map;
using std::string;

其作用域从using声明点开始,直到包含该using声明的作用域的末尾。
可以将较短的同义词与命名空间名字相关联,如:

namespace cplusplus_primer{ /* ... */ }
namespace primer = cplusplus_primer;
namespace Qlib = cplusplus_primer::QueryLib;

using指示使用如下:
using namespace std;
using指示将命名空间成员提升到包含命名空间本身和using指示的最近作用域。例如:

namespace A{
	int i, j;
}
void f(){
	using namespace A;	//injects names from A into the global scope.
	cout<<i*j<<endl;
	//...
}

 

类、命名空间和作用域
考虑下面的简单程序:

std::string s;
//calls std::getline(std::istream&, const std::string& )
getline(std::cin, s);

接受类类型形参(或类类型指针及引用形参)的函数(包含重载符),以及与类本身定义在同一命名空间中的函数(包括重载函数),在用类类型对象(或类类型的引用及指针)作为实参的时候是可见的。
当编译器在看到getline函数的使用getline(std::cin,s)的时候,它在当前作用域、包含调用的作用域以及定义cin的类型和string类型的命名空间中查找匹配的函数。因此它在std空间中查找并找到getline函数。

"当一个类声明友元函数的时候,函数的声明不必是可见的。如果不存在可见的声明,那么,友元声明具有将该函数或类的声明放入外围作用域的效果。如果类在命名空间内部定义,则没有另外声明的友元函数在同一命名空间中声明。"
这段话我读了很多遍,也不是很明白它的意思。友元声明将已命名的类或非成员函数引入到外围作用域中,意思应该是说,友元虽然在类内部声明或定义,但是他的作用域跟类的作用域一样(我的意思是说,它可以使用类的私有成员,我们可以像使用类一样在外围作用域使用函数)。如果我们在命名空间内定义类,那么友元函数也应该在同一命名空间声明,那么,这个“没有另外声明的友元函数”中,“没有另外声明”是什么意思呢?

 

命名空间对函数匹配有两个影响,一是using声明或using指示可以将函数加到候选集合。另外一种,有一个或多个类类型形参的函数的名字查找包括定义每个形参类型的命名空间,为查找候选函数而查找定义形参类(以及定义其基类)的每个命名空间,将那些命名空间中任意与被调用函数名字相同的函数加入候选集合。例如:

namespace NS{
	class Item_base{ /*... */ }
	void display( const Item_base& ){}
}

class Bulk_item : public NS::Item_base{ }

int main(){
	Bulk_item book1;
	display(book1);
	return 0;
}

display函数的实参book1具有类类型Bulk_item。display调用的候选函数不仅是在调用display函数的地方其声明可见的函数,还包括声明 Bulk_item类及其基类Item_base的命名空间中的函数。

 

3. 多重继承与虚继承
多重继承跟一般的继承没有太大区别,然而多个基类可能导致二义性。例如不同基类同时含有相同名字的成员,此时不论该成员是否可见,都会能查找名字,然后再判断声明的合法性。例如下面的代码:

struct Base1{
	Base1():ival(-11),dval(0), cval('b'){}
	void print( int )const;
	protected:
		int ival;
		double dval;
		char cval;
	private:
		int *id;
	};

struct Base2{
	Base2(){}
	void print( double )const;
	protected:
		double fval;
	private:
		double dval;
	};

struct Derived : public Base1{
	Derived():dval(0),sval("eva"){}
	void print( std::string ) const;
	protected:
		std::string sval;
		double dval;
	};
	
struct MI: public Derived, public Base2{
	MI(){
		ival = new int(11);
		dvec.push_back(1);
		dvec.push_back(2);
	}
	void print( std::vector<double> );
	void bar();
	void foobar( double );
	protected:
		int *ival;
		std::vector<double> dvec;
		};
		
void MI::bar(){
	int sval;
	//dval = 3.14;		//此时编译出错,dval在Derived及Base2中均有定义
	//虽然dval在Base2中是私有成员,然而编译器先找到了这两个匹配的声明,
	//已经出现了歧义。
	cval = 'a';
	//id = 1;
	fval = 0;
	sval = *ival;
}

虚继承是一种机制,类通过虚继承指出它希望共享其虚基类的状态。在虚继承下,对给定虚基类,无论该类在派生层次中作为虚基类出现多少次,只继承一个共享的基类子对象。
非虚继承的时候,我们通过向上逐步初始化各基类实现继承类的初始化,然而虚继承由于只继承一个共享的基类子对象,这使继承类的初始化变得复杂了起来:
在虚派生中,由最低层派生类的构造函数初始化虚基类。
无论虚基类出现在继承层次的任何地方,总是在构造非虚基类之前构造虚基类。

分享到:
评论

相关推荐

    ChromeOS镜像文件.zip

    目录: ChromeOS-PC-20130222-oscome.com ChromeOS-Vanilla-4028.0.2013_04_20_1810-r706c4144 ChromeOS-Vanilla-4028.0.2013_04_20_1810-r706c4144-VirtualBox ChromeOS-Vanilla-4028.0.2013_04_20_1810-r706c4144-VMWare ChromeOS-virtualbox-20130222-OSCOME.COM ChromeOS-vmware-20130222-OSCOME.COM 网盘文件永久链接

    ieee33节点matlab模型

    IEEE33节点模型搭建,matlab

    3GPP R15 38.331 5G NR无线资源控制(RRC)协议规范解析

    3GPP R15 38.331 5G NR无线资源控制(RRC)协议规范解析

    基于ssm+mysql实现的零食商城系统(电商购物).zip(毕设&课设&实训&大作业&竞赛&项目)

    项目工程资源经过严格测试运行并且功能上ok,可实现复现复刻,拿到资料包后可实现复现出一样的项目,本人系统开发经验充足(全栈全领域),有任何使用问题欢迎随时与我联系,我会抽时间努力为您解惑,提供帮助 【资源内容】:包含源码+工程文件+说明等。答辩评审平均分达到96分,放心下载使用!可实现复现;设计报告也可借鉴此项目;该资源内项目代码都经过测试运行,功能ok 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 【提供帮助】:有任何使用上的问题欢迎随时与我联系,抽时间努力解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 下载后请首先打开说明文件(如有);整理时不同项目所包含资源内容不同;项目工程可实现复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用,资源为网络商品(电子资料类)基于网络商品和电子资料商品的性质和特征不支持退款

    19考试真题最近的t44.txt

    19考试真题最近的t44.txt

    JSP基于SSH2新闻发布系统.zip(毕设&课设&实训&大作业&竞赛&项目)

    项目工程资源经过严格测试运行并且功能上ok,可实现复现复刻,拿到资料包后可实现复现出一样的项目,本人系统开发经验充足(全栈全领域),有任何使用问题欢迎随时与我联系,我会抽时间努力为您解惑,提供帮助 【资源内容】:包含源码+工程文件+说明等。答辩评审平均分达到96分,放心下载使用!可实现复现;设计报告也可借鉴此项目;该资源内项目代码都经过测试运行,功能ok 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 【提供帮助】:有任何使用上的问题欢迎随时与我联系,抽时间努力解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 下载后请首先打开说明文件(如有);整理时不同项目所包含资源内容不同;项目工程可实现复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用,资源为网络商品(电子资料类)基于网络商品和电子资料商品的性质和特征不支持退款,质量优质,放心下载使用

    19考试真题最近的t49.txt

    19考试真题最近的t49.txt

    19考试真题最近的t61.txt

    19考试真题最近的t61.txt

    电动汽车充电站选址定容优化:基于MATLAB建模求解与成本最小化策略,电动汽车充电站选址定容优化:基于MATLAB的最优规划模型及初学者指南,电动汽车充电站的最优选址定容MATLAB程序 以规划期内充

    电动汽车充电站选址定容优化:基于MATLAB建模求解与成本最小化策略,电动汽车充电站选址定容优化:基于MATLAB的最优规划模型及初学者指南,电动汽车充电站的最优选址定容MATLAB程序 以规划期内充电站的总成本 (包括投资、运行和维护成本)和网损费用之和最小为目标,考虑了相关的约束条件,构造了电动汽车充电站最优规划的数学模型。 从34个位置中,选取7个充电站地址,进行选址优化 关键词:电动汽车;充电站;选址和定容 程序注释清晰,适合初学者学习 ,电动汽车; 充电站选址定容; MATLAB程序; 规划模型; 成本优化; 网损费用; 初学者学习; 程序注释清晰,基于MATLAB的电动汽车充电站选址定容优化程序:成本最小化与约束条件下的选址策略

    威纶通触摸屏图库模板程序:多尺寸适用,PS原文件可自由修改,便捷电气助手应用,威纶通触摸屏图库模板程序:多尺寸适用,PS原文件可自由修改,便捷电气助手应用,威纶通触摸屏图库模板程序(电气助手) 可直接

    威纶通触摸屏图库模板程序:多尺寸适用,PS原文件可自由修改,便捷电气助手应用,威纶通触摸屏图库模板程序:多尺寸适用,PS原文件可自由修改,便捷电气助手应用,威纶通触摸屏图库模板程序(电气助手) 可直接使用。 内附原图、PS原文件可自行修改 不同触摸屏,不同寸尺都可以使用 ,威纶通触摸屏; 图库模板程序; 电气助手; 直接使用; 原图; 修改; 兼容不同寸尺,威纶通触摸屏图库模板程序:电气助手,便捷编辑通用模板

    群辉引导7.2.2 最新 vmware workstation 已经帮忙转换好为vmdk文件 直接使用就可以

    修复 "保存'/opt/rr'的修改" 后 主菜单锁死问题. 修复 trivial 插件的语法错误. 修复 open-vm-tools 套件 缺失的 SOCKETS 驱动. 添加 vmtools 插件, 包含 qemu-ga & open-vm-tools. 4.1. 该插件会自动判断环境并启用对应的功能, 物理机也不用刻意删除该插件. 4.2. 新安装用户会默认选中, 升级用户如需要请手动添加该插件. 4.3. 如启用该插件, 请不要再在系统中安装套件. 修复 wireless 插件. 5.1. 修复 RR 下无线网络 IP 显示和刷新问题. 5.2. 修复 RR 下设置 SSID&PSK 后 DSM 下不驱动的问题. 5.3. 同步 RR 下的 SSID&PSK 到 DSM 下. 5.4. 修复 junior 模式下无线网络的支持, 已支持 无线网卡的 DSM 系统安装. (暂时不支持 intel 无线网卡) 5.5. wpa_supplicant.conf 文件位于引导盘第一个分区根目录, 纯无线环境可手动放置该文件后其启动引导.

    19考试真题最近的t66.txt

    19考试真题最近的t66.txt

    19考试真题最近的t37.txt

    19考试真题最近的t37.txt

    Arduino-Mega2560开发板-毕业设计

    Arduino_Mega2560开发板工程文件 包含 原理图 PCB图

    智能养猪系统的高精度称重算法及其Python实现(含详细可运行代码及解释)

    内容概要:本文详述了一种用于智能养猪的高精度称重系统设计及其实现方法,主要涵盖了卡尔曼滤波、数据采集与预处理、重量估算与存储等功能。文中提供了完整的Python代码示例和详细的代码解释,旨在减少噪声干扰并提高数据准确性。具体而言,通过对采集的数据进行卡尔曼滤波,去除异常值,并使用一定时间段内数据的平均值作为最终的体重估计。此外,还实现了一个简单的图形用户界面,能够实时显示称重数据和估计的重量。 适合人群:农业自动化领域的开发者和技术爱好者,尤其关注智能畜牧业的技术应用。 使用场景及目标:适用于智能养猪场的精准称重,提高养猪效率和管理水平,确保获取高精度、可靠的牲畜体重数据,帮助养殖场更好地管理饲养过程。同时,提供完整的源代码有助于相关人员理解和优化现有系统。 阅读建议:对于想要深入了解智能畜牧业相关技术的读者来说,可以通过本教程掌握从硬件接入、软件设计再到数据处理全流程的具体细节。重点关注各个关键算法的实现原理及其应用场景,从而为自己的项目带来启示与借鉴。

    基于SSM框架构建积分系统和基本商品检索系统(Spring+SpringMVC+MyBatis+Lucene+Redis+MAVEN).zip(毕设&课设&实训&大作业&竞赛&项目)

    项目工程资源经过严格测试运行并且功能上ok,可实现复现复刻,拿到资料包后可实现复现出一样的项目,本人系统开发经验充足(全栈全领域),有任何使用问题欢迎随时与我联系,我会抽时间努力为您解惑,提供帮助 【资源内容】:包含源码+工程文件+说明等。答辩评审平均分达到96分,放心下载使用!可实现复现;设计报告也可借鉴此项目;该资源内项目代码都经过测试运行,功能ok 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 【提供帮助】:有任何使用上的问题欢迎随时与我联系,抽时间努力解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 下载后请首先打开说明文件(如有);整理时不同项目所包含资源内容不同;项目工程可实现复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用,资源为网络商品(电子资料类)基于网络商品和电子资料商品的性质和特征不支持退款

    最新更新!!!地级市-产业链韧性数据(2006-2021年)

    ## 01、数据简介 产业链韧性是指在产业链部分环节出现问题或遭受内外部冲击时,产业链仍能保持其稳定性和动态平衡,迅速做出反应并恢复正常运转的能力。这种能力体现了产业链的复杂适应性,是其能够应对各种不确定性因素和破坏性事件的重要保障。 产业链韧性是保障产业链安全稳定运行的重要基础,对于提升产业竞争力、推动经济高质量发展具有重要意义。 数据名称:地级市-产业链韧性数据 数据年份:2006-2021年 ## 02、相关数据 代码 年度 城市 产业结构HHI 获得专利数 第一产业增加值占GDP比 第二产业增加值占GDP比 第三产业增加值占GDP比 产业链韧性

    PNP发射极接地开关仿真原理图

    PNP发射极接地开关仿真原理图

    上门预约服务小程序v4.10.9+前端.zip

    上门预约服务小程序v4.10.9+前端 文章列表单图时,图标统一左侧对齐 文章内增加视频位置,显示在文章顶部 文章内底部导航增加首页、分享、自定义按钮,可跳转内部页面、其他小程序、业务域名内的H5页面,方便宣传使用

    Python环境下的滚动轴承故障诊断优化算法:基于改进WDCNN的一维卷积神经网络与LSTM融合的时序信号处理研究,Python环境中基于改进WDCNN与LSTM融合的滚动轴承故障诊断方法研究-优化

    Python环境下的滚动轴承故障诊断优化算法:基于改进WDCNN的一维卷积神经网络与LSTM融合的时序信号处理研究,Python环境中基于改进WDCNN与LSTM融合的滚动轴承故障诊断方法研究——优化卷积核大小,提升诊断准确率并加速收敛速度的应用,Python环境下一种基于WDCNN的滚动轴承故障诊断方法 算法采用pytorch深度学习模块,对WDCNN进行改进,搭建了卷积核大小逐层递减的一维卷积神经网络,并减少了卷积层数量,达到了98%以上的诊断准确率,同时有着较快的收敛速度。 另外,针对时序信号的特点,将长短时记忆网络(LSTM)与搭建的一维卷积神经网络结合,提高分类准确率至99%以上,但收敛速度较单一的卷积神经网络较慢。 算法可迁移至金融时间序列,地震信号,语音信号,声信号,生理信号(ECG,EEG,EMG)等一维时间序列信号。 ,基于WDCNN的故障诊断方法; 卷积神经网络; 算法改进; 高诊断准确率; 收敛速度快; LSTM结合; 一维时间序列信号; 金融、地震、语音、生理信号诊断,Python下改进WDCNN的滚动轴承故障诊断法:深度学习提升诊断准确率与收敛速度

Global site tag (gtag.js) - Google Analytics