使用C++11实现完美资源管理

1.资源管理包括内存管理、文件句柄等等需要进行打开(申请)、关闭(释放)操作的过程

2.VS2010使用的C++规范,严格说来不是C++11,而是C++0x,但是一脉相承的

一:管理数组

相较于auto_ptr,unique_ptr的增强点之一是支持对数组对象指针的管理,比如:

struct A
{
    int m_data;
    A(int n=0){m_data =n;cout<<n<<" 被构造了"<<endl;}
    ~A(){cout<<m_data<<" 被销毁了"<<endl;}
    void Set(int n){m_data=n;
};

auto_ptr<A> p1(new A[5]);//由于自动释放时,只调用delete,所以会导致A[0]以后对象的析构函数不会被调用。对于需要在析构函数中释放资源的对象来说,是不能接受的。

unique_ptr<A[]> p2(new A[5]);//这样,在释放p2时,会使用delete [].

二:管理资源

在上一篇中,使用了sqlite:

sqlite3 *db = NULL;

//其他操作

sqlite3_close(db);

还有:

sqlite3_stmt *pstmt=NULL;

//其他操作

sqlite3_reset(pstmt);或者sqlite3_finalize(pstmt);

使用智能指针或其他wrapper类管理资源,主要是
1.方便省事,最重要的是,避免资源泄露、未释放

2.应对可能出现的异常.异常出现时,可以自动释放资源,这就是所谓异常安全编程的三个条件之一(能释放资源;能释放资源且保证资源在异常前后状态一致;不抛出异常).另一方面,也可以使得代码美观简洁,否则为对付异常,不得不使用大堆大堆的if…else…

下面具体来看:

借助于C++11(C++0x)新引入的auto、decltype、匿名函数对象(Lambda表达式)等特性,我们可以方便进行资源管理:

sqlite3 *pdb = NULL;

auto deleter = [](sqlite3 *pdb){sqlite3_close(pdb);}

int nRet =  sqlite3_open16(L"F:\\my.db",&db);

std::unique_ptr<sqlite3,decltype(deleter)> pdb(pdb,deleter);

if(nRet)

{//失败,但是这里不用担心资源管理了

}

else

{

}

//其他操作

//结束前,会动释放资源

这样,其他的各类资源:User对象HWND,GDI对象HBRUSH,Kernel对象HANDLE等,都可以用此方法进行资源管理,凡此种种,可以自由发挥。哈哈,妈妈再也不用担心我申请资源不释放了!

下一篇,也许可以谈谈借助于11里面的bind和function,利用stratgy模式,实现虚函数的效果。(Effective C++有个Item谈到了这种方法)

posted on 2014-04-16 21:23  qinfengxiaoyue  阅读(1218)  评论(0编辑  收藏  举报