zoukankan      html  css  js  c++  java
  • 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;
    }
    
  • 相关阅读:
    移动端开发常遇问题解答
    CSS属性小结之--半透明处理
    jquery事件代理
    sprite图在移动端的使用
    vertical-align及IE7下的inline-block
    图片延迟加载
    iOS NSInteger/NSUInteger与int/unsigned int、long/unsigned long之间的区别!
    GCD深入理解(1)
    iOS 开发图片资源选择png格式还是jpg格式
    iOS沙盒(sandbox)机制及获取沙盒路径
  • 原文地址:https://www.cnblogs.com/ltimaginea/p/15487346.html
Copyright © 2011-2022 走看看