effective C++ 读书笔记 条款08「建议收藏」

effective C++ 读书笔记 条款08

大家好,又见面了,我是全栈君。

条款08  别让异常逃离析构函数:

假设在析构函数其中发生了异常,程序可能会过早结束或者导致不明白行为(异常从析构函数传播出去)

 

看代码:

#include <iostream>
using namespace std;


class DBConnection
{
public:


	void close()
	{
			int i = 3;
			int j = 0;
			int k = i/j;
			printf("%d\n",k);
	}
};

class DBConn
{
public:
	DBConn()
	{

	}

	/*
	~DBConn()
	{
			db.close();
	}
	*/



	//解决的方法1:强迫结束程序
	
	~DBConn()
	{
		try
		{
			db.close();
		}
		catch(...)
		{
			abort();//假设一个程序遭遇一个“于析构期间发生的错误”后无法继续同意,强迫结束是个合理的选择。用abort阻止异常从析构函数传播出去
		}
	}
	


	//解决的方法2:吞下异常
	/*
	~DBConn()
	{
		try
		{
			db.close();
		}
		catch (...)
		{
			//制作运转记录,记下对close的调用失败
		}
	
	}
	*/
private:
	DBConnection db;
};


int main()
{
	DBConn dbc;
	//dbc对象销毁时会自己主动调用DBConnection的close函数;仅仅要调用close成功,一切美好,可是假设该调用出现异常。DBConn析构函数
	//会传播该异常。也就是同意它离开这个析构函数。

会造成问题。

return 0;}/*採用本来的析构函数:执行程序。提示一个程序已经停止工作(感觉这里编译器优化到解决方式1了,自己主动做了处理),假设不停止工作。也出现了不明白行为。

採用解决方式1:执行程序,提示一个debug error,终止程序。採用解决方式2:执行程序,程序执行通过。可是也不会打印k的值*/

上面的解决方式1和2。都导致对close抛出的异常无法处理:

最佳办法就是DBConn自己又一次设计一个close接口,使得程序猿能够自己调用:

#include <iostream>
using namespace std;


class DBConnection
{
public:


	void close()
	{
			int i = 3;
			int j = 0;
			int k = i/j;
			printf("%d\n",k);
	}
};

class DBConn
{
public:
	DBConn()
	{

	}
	~DBConn()
	{
		if (!closed)
		{
			try  //关闭连接。假设客户不那么做的话
			{
				db.close()
			}
			catch(...)
			{
				//在这里结束程序或者吞下异常
			}
		}

	}

	void close()
	{
		db.close();
		closed = true;
	}


private:
	DBConnection db;
	bool closed;
};


int main()
{
	DBConn dbc;
	//dbc对象销毁时会自己主动调用DBConnection的close函数;仅仅要调用close成功,一切美好,可是假设该调用出现异常,DBConn析构函数
	//会传播该异常,也就是同意它离开这个析构函数。会造成问题。


	return 0;
}

/*

这种方法把调用close的责任从DBConn析构函数手上移动到DBConn客户手上(但DBConn析构函数仍含有一个“双保险”调用).

假设某个操作可能在失败时抛出异常。而又存在某种须要必须处理该异常,那么这个异常必须来自析构函数以外的某个函数
由于析构函数吐出异常,总会带来“过早结束程序”或者“发生不明白行为”的风险。

这里由客户自己调用close函数。假设发现异常,能够处理。总结: 1:析构函数绝对不要吐出异常,假设一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉该异常,然后吞下它们(不传播)或结束程序 2:假设客户须要对某个操作函数执行期间抛出的异常做出反应,那么class应该提供一个普通函数(而非析构函数)执行该操作。*/

 

1:析构函数绝对不要吐出异常。假设一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉该异常。然后吞下它们(不传播)或结束程序

2:假设客户须要对某个操作函数执行期间抛出的异常做出反应,那么class应该提供一个普通函数(而非析构函数)执行该操作。
 

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/115238.html原文链接:https://javaforall.net

(0)
上一篇 2022年2月7日 上午10:00
下一篇 2022年2月7日 上午11:00


相关推荐

  • 随机森林算法及其实现(Random Forest)

    阅读目录1什么是随机森林? 2随机森林的特点 3随机森林的相关基础知识 4随机森林的生成 5袋外错误率(ooberror) 6随机森林工作原理解释的一个简单例子 7随机森林的Python实现 8参考内容回到顶部1什么是随机森林?  作为新兴起的、高度灵活的一种机器学习算法,随机森林(RandomForest,简称RF)拥有广泛的应用前景,从市场营销…

    2022年4月9日
    72
  • java 缓冲区溢出_缓冲区溢出详解

    java 缓冲区溢出_缓冲区溢出详解1 缓冲区溢出原理缓冲区是一块连续的计算机内存区域 可保存相同数据类型的多个实例 缓冲区可以是堆栈 自动变量 堆 动态内存 和静态数据区 全局或静态 在 C C 语言中 通常使用字符数组和 malloc new 之类内存分配函数实现缓冲区 溢出指数据被添加到分配给该缓冲区的内存块之外 缓冲区溢出是最常见的程序缺陷 栈帧结构的引入为高级语言中实现函数或过程调用提供直接的硬件支持 但由于将函数返回地址

    2026年3月18日
    1
  • Linux命令 – ls命令

    Linux命令 – ls命令Linux 命令 ls 命令 ls 命令是 linux 下最常用的命令 ls 命令就是 list 的缩写 用来打印出当前目录的清单 如果 ls 指定其他目录 那么就会显示指定目录里的文件及文件夹清单 通过 ls 命令不仅可以查看 linux 文件夹包含的文件 而且可以查看文件权限 包括目录 文件夹 文件权限 查看目录信息等等 1 语法 ls 选项 目录或者文件 2 功能 列出目录的内容 包括文件和子目录的名称 3 参数 1 每列仅显示一个文件或目录名称 a 或 all 下所有文件和目录 A

    2025年11月16日
    5
  • 电脑上怎么看用户名和密码_电脑超级用户名和密码

    电脑上怎么看用户名和密码_电脑超级用户名和密码大家好,我是时间财富网智能客服时间君,上述问题将由我为大家进行解答。以windows7为例,电脑的用户名和密码的查看方法是:1、找到“我的电脑”或计算机;2、单击鼠标右键,打开“管理”;3、点击“本地用户和组”;4、选择用户,单击打开;5、查看用户及密码,也可以开机启动Windows,当运行到“正在启动WindowsXP”的提示界面时,按“F8”键调出系统启动选择菜单,选择“带命令行安全模式”;…

    2022年10月11日
    4
  • python安装失败0x80070570_win7系统出现windows无法安装错误代码0x80070570的解决方法…

    python安装失败0x80070570_win7系统出现windows无法安装错误代码0x80070570的解决方法…无论谁在使用电脑的时候都可能会发现出现 windows 无法安装错误代码 0x 的问题 出现 windows 无法安装错误代码 0x 让用户们很苦恼 这是怎么回事呢 出现 windows 无法安装错误代码 0x 有什么简便的处理方式呢 其实只要依照 1 原因可能是分区工具的快速处理方法 分区出问题导致安装文件不认识分区 所以出现上面的 安装程序无法创建新的分区或者无法识别

    2026年3月19日
    1
  • 动态规划之最长回文子串

    动态规划之最长回文子串问题:给出一个字符串S,求S的最长回文子串的长度。样例字符串”PATZJUJZTACCBCC”的最长回文子串为”ATZJUJZTA”,长度为9。        还是先看暴力解法:枚举子串的两个端点i和j,判断在[i,j]区间内的子串是否回文。从复杂度上来看,枚举端点需要0(n2),判断回文需要0(n),因…

    2022年4月26日
    39

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号