new 捕获所有异常 避免内存泄漏

2021/10/31 7:10:33

本文主要是介绍new 捕获所有异常 避免内存泄漏,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

原始指针 捕获所有异常 避免内存泄漏

原始指针 避免内存泄漏

程序使用 raw pointer 时,资源往往被显式管理(managed explicitly)。以此方式使用 raw pointer 的典型例子是,以 newdelete 创建和销毁对象:

void Foo()
{
	Widget* ptr = new Widget;	// create an object explicitly

	// ... perform some operations(may throw exceptions)

	delete ptr;			// clean up(destroy the object explicitly)
}

这个函数将成为麻烦制造者。一个明显的问题是,有可能忘记 delete 对象,特别是如果你在函数中有个 return 语句。另一个较不明显的危险是它可能抛出异常,那将立刻退离(exit)函数,末尾的 delete 语句也就没机会被调用,导致内存泄漏,或更一般性地说,资源泄漏。

为了避免如此的资源泄漏,通常函数会捕捉所有异常,例如:

void Foo()
{
	Widget* ptr = new Widget;

	try
	{
		// perform some operations...(may throw exceptions)
		std::string().at(1);	// this generates an std::out_of_range
	}
	catch (...)			// for any exception
	{
		delete ptr;		// clean up
		throw;			// rethrow the exception
	}

	delete ptr;			// clean up on normal end
}

为了在异常发生时能够适当处理好对象的删除,代码变得比较复杂,而且累赘。如果第二个对象也以此方式处理,或需要一个以上的 catch 子句,情况会变得更糟。这是一种不好的编程风格,应该避免,因为它复杂而又容易出错。 对此, smart pointer 可以带来帮助。 Smart pointer 可以在它自身被销毁时释放其所指向的数据——不论是正常结束或异常结束。现代 C++ 程序应该优先使用智能指针。

完整程序

#include <iostream>
#include <new>

class Widget { double d[1024]; /* ... */ };

void Foo()
{
	Widget* ptr = new Widget;

	try
	{
		// perform some operations...(may throw exceptions)
		std::string().at(1);	// this generates an std::out_of_range
	}
	catch (...)			// for any exception
	{
		delete ptr;		// clean up
		throw;			// rethrow the exception
	}

	delete ptr;			// clean up on normal end
}

int main()
{
	try
	{
		Foo();
	}
	catch (const std::bad_alloc& err)
	{
		std::cout << err.what() << std::endl;
	}
	catch (const std::exception& err)
	{
		std::cout << err.what() << std::endl;
	}
	catch (...)
	{
		std::cout << "unknown exceptions" << std::endl;
	}

	return 0;
}


这篇关于new 捕获所有异常 避免内存泄漏的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程