More Effective C++ 条款09:利用析构函数避免资源泄漏

本文探讨了在C++中如何通过异常处理确保资源的有效管理,特别是在处理可能抛出异常的函数调用时。通过使用智能指针如`std::shared_ptr`,可以确保在异常发生时正确地释放对象。此外,还介绍了创建一个名为`WindowHandle`的类,用于管理窗口资源,即使在显示信息过程中发生异常,也能确保窗口资源的正确释放。这展示了在设计中如何将异常安全性和资源管理结合起来。
摘要由CSDN通过智能技术生成
class ALA
{
public:
    virtual void processAdoption() = 0;
};

class Puppy : public ALA
{
public:
    virtual void processAdoption();
};

class Kitten: public ALA
{
public:
    virtual void processAdoption();
};

//核心函数
void processAdoptions(istream& dataSource)
{
    while(dataSource){
        ALA *pa = readALA(dataSource);  //从流中读取动物,新分配对象,有着适当的类型(Puppy,Kitten)
        pa->processAdoption();
        delete pa;
    }
}

现在考虑,如果pa->processAdoption抛出一个异常,会发生什么事情?

processAdoptions无法捕捉到它,所以异常会抛出到processAdoptions的调用端,processAdoptions函数内位于pa->processAdoption之后的所有语句都会被跳过,不再执行,这意味着pa不会被删除。

解决办法是,以一个“类似指针的对象”取代指针pa,如此一来,当这个类似指针的对象被“自动”销毁,我们可以令其析构函数调用delete。行为类似指针的对象我们称为智能指针。可以用C++智能指针shared_ptr。以下为改写后的代码

void processAdoptions(istream& dataSource)
{
    while(dataSource){
        std::shared_ptr<ALA> pa (readALA(dataSource));  //从流中读取动物,新分配对象,有着适当的类型(Puppy,Kitten)
        pa->processAdoption();
    }
}

考虑图形界面应用软件中的某个函数,它必须产生一个窗口显示某些信息:

void displayInfo(const Information &info)
{
    WINDOW_HANDLE w(createWindow());
    display info in window corresponding to w;
    destoryWindow(w);
}

如果在信息显示于w的过程中发生exception,w所持有的那个窗口将会遗失,其他动态分配的任何资源也会遗失。解决之道和先前一样,设计一个class,令其构造和析构分别取得资源和释放资源。

class WindowHandle
{
public:
    WindowHandle(WINDOW_HANDLE handle) : w(handle){}
    ~WindowHandle() {destroyWindow(w);}
    
    WindowHandle(const WindowHandle&) = delete;
    WindowHandle& operator=(const WindowHandle&) = delete;
    operator WINDOW_HANDLE() { return w; }  //类型转换操作

private:
    WINDOW_HANDLE w;
};

有了这个WindowHandle class,我们可以重写displayInfo:

void displayInfo(const Information& info)
{
    WindowHandle w(createWindow());
    display info in window corresponing to w;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值