[C++ Effective]第3章 资源管理

条款13:以对象管理资源

void f(){
    InvestMent* pInv = createInvestment();
    ...
    delete pInv;
}

对于上述情况,可能存在很多时候无法调用delete进行资源的释放。

1、过早的遇到一个return语句;

2、delete在循环中提前遇到break或continue;

3、遇到抛出异常的情况。

因此为了确保createInvestment返回的资源总是被释放,需要将资源放进对象内,当控制流离开函数f,该对象的析构函数会自动释放那些资源。

例如可以使用auto_ptr指针,其析构函数自动对其所指对象调用delete:

void f(){
    std::auto_ptr<Investment> pInv(createInvestment());
    ...
}

auto_ptr定义的对象,若通过copy构造函数或copy assignment操作符来复制他们,他们会变成null,而复制所得的指针将取得资源的唯一拥有权。因此auto_ptr无法完成正常的赋值与复制操作。

解决方案是使用shared_ptr指针,使用方法类似,但是shared_ptr存在一个计数器用于追踪共有多少对象执行资源,并在无人指向它时自动删除该资源。

但是shared_ptr和auto_ptr是 使用delete进行释放,不是delete[],因此不要在动态分配得到的array身上使用shared_ptr和auto_ptr,如:

std::auto_ptr<std::string> aps(new string[10]);
std::shared_ptr<int> spi(new int[10]);

条款14:在资源管理类中小心copying行为

当一个对象被复制时,处理方式:

1、禁止复制。即吧复制构造函数及copying assignment设置为private。

2、对底层资源祭出“引用计数法”,通过只要内含一个shared_ptr成员变量,便可实现出引用计数法。甚至可以自定义删除器,而不是使用delete

class Lock{
public:
    explicit Lock(Mutex* pm): mutexPtr(pm, unlock){    //指定unlock为删除器
        lock(mutexPtr.get());
    }

private:
    std::str1::shared_ptr<Mutex> mutexPtr;
};

mutexPtr的析构函数会在互斥器的引用次数为0时自动调用删除器unlock

3、复制底层资源。即进行深度拷贝

4、转移底部资源的拥有权。即使用auto_ptr

条款15:在资源管理类中提供对原始资源的访问

shared_ptr 和 auto_ptr都提供了get函数来获得原始指针,也都对*和->进行了操作符重载

对于获得底部资源,可以提供两种方法:

1、使用get()函数获得对象的底部资源。

2、使用隐式转换函数

class Font{
    FontHandle f;
public:
    explicit Font(FontHandle fh):f(fh){}
    operator FontHandle()const{ return f;}
}

int main(){
    Font f(getFont());
    changeFontSize(f);        //将Font隐式转换为FontHandle
}

 条款16:成对使用new和delete时要采用相同形式

new和delete以及new []和delete[]是配对的。

条款17:以独立语句将newed对象置于智能指针

shared_ptr的构造函数是explicit声明的,不能直接使用隐式装换,但可以强制转换,如

int priority();
void processWidget(shared_ptr<Widget> pw, int priority);
//在调用processWidget时有:
processWidget( shared_ptr<Widget>(new Widget()), priority)

上述processWidget调用方式可能造成资源泄露,C++没有明确参数核算的特定顺序,因此上述调用可能分为以下三部分组成:

1、执行new Widget

2、调用priority

3、调用shared_ptr

但是在调用priority时可能出现异常,导致new Widget无法使用shared_ptr进行释放,因此解决方案就是以独立语句将newed对象置入智能指针

即:

int priority();
void processWidget(shared_ptr<Widget> pw, int priority);
//在调用processWidget时有:
shared_ptr<Widget> pw(new Widget());
processWidget( pw, priority);
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值