C++中异常保证资源释放的正确用法
acmilan2016/11/26软件综合 IP:四川

如果正在使用C++/CLI

C++/CLI已经有了try-finally,C++异常在.NET中被包装为SEHException,因此try-finally可以正常处理这些情况。

特别是ref class一定要使用到try-finally,因为ref class的析构函数需要使用delete运算符手动调用,系统是不会自动调用ref class的析构函数的。

<code class="language-cpp">// CLRConsoleApp1.cpp: 主项目文件。

#include "stdafx.h"

using namespace System;

ref class MyClass
{
public:
	MyClass()
	{
		Console::WriteLine(L"MyClass()");
	}
	~MyClass()
	{
		Console::WriteLine(L"~MyClass()");
	}
};

int main(array<system::string ^> ^args)
{
	MyClass ^obj = gcnew MyClass();
    try
	{
		// ...
	}
	finally
	{
		delete obj;
	}
    return 0;
}
</system::string></code>

如果正在使用纯C语言编写函数

可以使用__try-__finally这一组SEH,不过需要使用__leave正常离开这个范围避免回卷。这个功能是不允许在存在像std::string等C++栈对象的环境中使用的,否则会报错。

由于这种方法限制较多,不实用,因此并不推荐。

如果正在使用传统C++

传统C++中,一般使用析构函数保证异常资源释放,一般称之为RAII。

个人曾经比较反对RAII,原因是当时认为使用RAII意味着包装类型,而包装类型之后将会降低可迁移性。但其实使用RAII并不意味着非得包装类型,也可以只包装赋值和释放操作。

<code class="language-cpp">// main.cpp

#include <stdio.h>

struct StdioFileMgr
{
	FILE *&f;
	StdioFileMgr(FILE *&f, const char *fn, const char *md) : f(f)
	{
		f = fopen(fn, md);
	}
	StdioFileMgr(FILE *&f, const wchar_t *fn, const wchar_t *md) : f(f)
	{
		f = _wfopen(fn, md);
	}
	~StdioFileMgr()
	{
		if (f != NULL) fclose(f);
		f = NULL;
	}
};

int main()
{
	{
		FILE *f1 = NULL;
		StdioFileMgr f1mgr(f1, "xxx.txt", "r");
		// ...
	}

	return 0;
}
</stdio.h></code>

[修改于 7年4个月前 - 2016/12/31 18:20:16]

来自:计算机科学 / 软件综合
1
已屏蔽 原因:{{ notice.reason }}已屏蔽
{{notice.noticeContent}}
~~空空如也
acmilan 作者
7年5个月前 修改于 7年4个月前 IP:四川
827947

不要包装类型,因为会降低可迁移性。应该包装赋值和释放操作,保留原始C语言类型。

类似地,Win32文件打开操作可以这样包装。

<code class="language-cpp">struct Win32FileMgr
{
	HANDLE &f;
	Win32FileMgr(HANDLE &f, LPCTSTR fn, DWORD da, DWORD sm,
		LPSECURITY_ATTRIBUTES sa, DWORD cd, DWORD fa, HANDLE ht) : f(f)
	{
		f = CreateFile(fn, da, sm, sa, cd, fa, ht);
	}
	~Win32FileMgr()
	{
		if (f != NULL && f != INVALID_HANDLE_VALUE) CloseHandle(f);
		f = INVALID_HANDLE_VALUE;
	}
};
</code>

其实这个用法不仅可以用于资源的情况,实际上可以用于所有【必须成对进行的操作】的包装。不光可以用于因为异常跳出的情况,实际上可以用于在任何【多出口的函数】中实现对成对操作的包装。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论

想参与大家的讨论?现在就 登录 或者 注册

所属专业
所属分类
上级专业
同级专业
acmilan
进士 学者 笔友
文章
461
回复
2934
学术分
4
2009/05/30注册,5年2个月前活动
暂无简介
主体类型:个人
所属领域:无
认证方式:邮箱
IP归属地:未同步
文件下载
加载中...
{{errorInfo}}
{{downloadWarning}}
你在 {{downloadTime}} 下载过当前文件。
文件名称:{{resource.defaultFile.name}}
下载次数:{{resource.hits}}
上传用户:{{uploader.username}}
所需积分:{{costScores}},{{holdScores}}下载当前附件免费{{description}}
积分不足,去充值
文件已丢失

当前账号的附件下载数量限制如下:
时段 个数
{{f.startingTime}}点 - {{f.endTime}}点 {{f.fileCount}}
视频暂不能访问,请登录试试
仅供内部学术交流或培训使用,请先保存到本地。本内容不代表科创观点,未经原作者同意,请勿转载。
音频暂不能访问,请登录试试
支持的图片格式:jpg, jpeg, png
插入公式
评论控制
加载中...
文号:{{pid}}
投诉或举报
加载中...
{{tip}}
请选择违规类型:
{{reason.type}}

空空如也

加载中...
详情
详情
推送到专栏从专栏移除
设为匿名取消匿名
查看作者
回复
只看作者
加入收藏取消收藏
收藏
取消收藏
折叠回复
置顶取消置顶
评学术分
鼓励
设为精选取消精选
管理提醒
编辑
通过审核
评论控制
退修或删除
历史版本
违规记录
投诉或举报
加入黑名单移除黑名单
查看IP
{{format('YYYY/MM/DD HH:mm:ss', toc)}}