Effective C++条款13、14

条款十三:以对象管理资源

从这个条款开始,就进入到资源管理部分了。资源管理往往是大型项目的一个难点,也是重中之重,看到一些编程规范,都是将资源管理的规范列为高优先级的重点。

管理资源的最好方法其实是预防,而好的预防方法就是尽量不去使用C/C++的原生指针,这些指针像幽灵一样,一个“忘记”,就是一个隐患。当项目小的时候,这些隐患看不出来,但当研发一个拥有上万级用户的产品时,服务器对很多人同时运行含有隐患的代码,这个隐患就会爆发,导致内存不足而崩溃。

 int *p = new int();
 …
 delete p;

这个当然是OK的,内存可以回收,但万一中间的过程超过数十行,你还能记住去delete p吗?
有些同学可能会这样做,每一个new之后,先写好delete,再往中间插代码,但如果遇到这样的情况:

 int* GetResource()
 {
  return new int(10);
 }

这个函数负责生成资源,并将之返回给上一层,因为这个资源是要在上一层用的,所以在函数内还不能直接delete,这可就麻烦了。程序员们可要随时记挂着这个内存,在不再需要的时候赶紧释放了。但一切可能出错的地方真的会出错,代码一多,记性就不那么好了,很容易出现忘记delete的事情。

那么有没有什么好的方法,可以让我们痛快地管理资源呢?这就牵涉到智能指针的问题了。常用的智能指针有memory头文件里面的auto_ptr,还有boost库里的shared_ptr。两者的实现机制不同,但功能是类似的,就是能自动管理资源。他们体现的思想就是RAII(Resources Acquisition Is Initialzeation 资源取得的时机便是初始化时机)。
举个形象的例子就是:

auto_ptr ap(new int(10));

或者是:

shared_ptr sp(new int(10));

可以看到,这里根本就没有出现原生指针,而是直接将new出来的内存交给了一个“管理者”。这个管理者可以是auto_ptr,也可以是shared_ptr,他们的存在,使得程序员可以真的忘记delete了,当ap或者sp的生命周期结束时,他们会将资源释放出来,交还给系统。比如:

void fun()
{
auto_ptr<int> ap(new int(10));
*ap = 20;
}
// ap 在fun()结束后会自动回收掉

auto_ptr和shared_ptr的区别在于管理方法不同。

auto_ptr这个管理者是一个很霸气的boss,他只想独管资源,而不允许其他管理者插手,否则他就退出管理,把这个资源交给另一方,自己再也不碰了。

举个例子:

auto_ptr<int> boss1(new int(10)); // 这个时候boss1对这个资源进行管理

auto_ptr<int> boss2(boss1); 
// 这个时候boss2要插手进来,boss1很生气,所以把管理权扔给了boss2,自己就不管事了。所以boss2持有这个资源,而boss1不再持有(持有的是null)。

所以同一时刻,只有一个auto_ptr能管理相同的资源。
这里还是有必要解释一下这句话的,比如:

 auto_ptr<int> boss1(new int(10)); 
 auto_ptr<int> boss2(new int(10));

boss1还持有资源吗?答案是有的,因为new执行了两次,虽然内存空间里的初始值是一样的,但地址并不同,所以不算相同的资源。
再来,为了把问题讲清楚,这里就让原生指针再出场一下

 int *p = new int(10);
 auto_ptr<int> boss1(p);
 auto_ptr<int> boss2(p);

boss1还持有资源吗?
答案是当程序运行到第三句的时候,就弹出assertion failed的红叉,程序崩溃了,为什么会这样呢?因为p所指向的是同一块资源,所以boss2发出管理手段后,boss1肯定要有动作的。第三句话调用boss2的构造函数,但构造函数中boss2识别出来p所指向的资源已经被占用了,所以会assertion failed。

事实上,这种原生指针和智能指针混用的程序是非常不好的,如果不使用原生指针,上面的崩溃问题就不会发生。
根据auto_ptr的思想,仿写了一下这个类,有兴趣的读者可以看看,其实并不复杂:

#ifndef MY_AUTO_PTR_H
#define MY_AUTO_PTR_H

// 自定义的智能指针
#include <iostream>
using namespace std;

template <class T>
class MyAutoPtr
{
private:
    T *ptr;
    static void swap(MyAutoPtr &obj1, MyAutoPtr &obj2)
    {
        // 细到交换成员变量
        std::swap(obj1.ptr, obj2.ptr);
    };

public:
    explicit MyAutoPtr(T *p = NULL): ptr(p){}

    MyAutoPtr(MyAutoPtr& p)
    {
        ptr = p.ptr;
        p.ptr = 0;
    }

    MyAutoPtr& operator= (MyAutoPtr& p)
    {
        if(&p != this)
        {
            MyAutoPtr temp(p);
            swap(*this, temp);
        }
        return *this;
    }

    ~MyAutoPtr()
    {
        delete ptr;
        ptr = 0;
    }

    T& operator* () const
    {
        return *ptr;
    }

    T* operator-> () const 
    {
        return ptr;
    }

    T* get() const
    {
        return ptr;
    }


    friend ostream& operator<< (ostream& out, const MyAutoPtr<T>& obj)
    {
        out << *obj.ptr;
        return out;
    }

};


#endif /* MY_AUTO_PTR_H */

MyAutoPtr

因为auto_ptr常常发生管理权的交付,所以用作形参的时候一定要小心:
若有一个函数是:

void fun(auto_ptr<int> ap){}

在主函数中调用fun(main_ap);
main_ap就会失去管理权了。
正是因为这个特性,所以auto_ptr不支持STL容器,因为容器要求“可复制”,但auto_ptr只能有一个掌握管理权。另外,auto_ptr也不能用于数组,因为内部实现的时候,用的是delete,而不是delete[]。

shared_ptr则顾名思义,他是一个分享与合作的管理者,他的内部实现是引用计数型的,每多一次引用,计数值就会+1,而每一次引用的生命周期结束时,计数值就会-1,当计数值为0的时候,说明内存没有管理者管理了,最后一个管理这个内存的管理者就会将之释放。

再回到我们之前的例子:

 shared_ptr<int> boss1(new int(10)); // 这个时候boss1对这个资源进行管理
 shared_ptr<int> boss2(boss1); // boss2携手boss1对这个资源进行管理

注意这个时候boss1并没有交出管理权,boss2的加入只对对象内部的计数指针造成了影响,在外部就像什么也没发生一样,可以使用boss1和boss2来管理资源。

下面写上我自己写的shared_ptr,也很简单的(注意swap函数要自己去写,不能调用系统自带的,否则会陷入赋值运算符的无限递归中):

#ifndef MY_SHARED_PTR_H
#define MY_SHARED_PTR_H

#include <iostream>
using namespace std;

template <class T>
class MySharedPtr
{
private:
    T *ptr;
    size_t *count;
    static void swap(MySharedPtr& obj1, MySharedPtr& obj2)
    {
        std::swap(obj1.ptr, obj2.ptr);
        std::swap(obj1.count, obj2.count);
    }

public:
    MySharedPtr(T* p = NULL): ptr(p), count(new size_t(1)){}

    MySharedPtr(MySharedPtr& p): ptr(p.ptr), count(p.count)
    {
        ++ *p.count;
    }

    MySharedPtr& operator= (MySharedPtr& p)
    {
        if(this != &p && (*this).ptr != p.ptr)
        {
            MySharedPtr temp(p);
            swap(*this, temp);
        }
        return *this;
    }

    ~MySharedPtr()
    {
        reset();        
    }

    T& operator* () const
    {
        return *ptr;
    }

    T* operator-> () const 
    {
        return ptr;
    }

    T* get() const 
    {
        return ptr;
    }

    void reset()
    {
        -- *count;
        if(*count == 0)
        {
            delete ptr;
            ptr = 0;
            delete count;
            count = 0;
            //cout << "真正删除" << endl;
        }
    }

    bool unique() const
    {
        return *count == 1;
    }

    size_t use_count() const 
    {
        return *count;
    }


    friend ostream& operator<< (ostream& out, const MySharedPtr<T>& obj)
    {
        out << *obj.ptr;
        return out;
    }

};

#endif /* MY_SHARED_PTR_H */

MySharedPtr

好,最后总结一下:

  1. 为了防止资源泄漏,使用RAII思想,它们在构造函数中获得资源,并在析构函数中释放资源。

  2. 两个常用的RAII类是shared_ptr和auto_ptr,但前者一般是更佳的选择,因为其copy行为比较直观,若选择auto_ptr,复制动作会使它指向null。

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

首先来看一个例子:

#include <iostream>
using namespace std;

class Lock
{
public:
    explicit Lock(int* pm): m_p(pm)
    {
        lock(m_p);
    }

    ~Lock()
    {
        unlock(m_p);
    }

private:
    int *m_p;
    void lock(int* pm)
    {
        cout << "Address = " << pm << " is locked" << endl;
    }

    void unlock(int *pm)
    {
        cout << "Address = " << pm << " is unlocked" << endl;
    }
};


int main()
{
    int m = 5;
    Lock m1(&m);
}

这个是模仿原书中的例子,做的一个加锁和解锁的操作。

运行结果如下:
这里写图片描述
这符合预期,当m1获得资源的时候,将之锁住,而m1生命周期结束后,也将资源的锁释放。

注意到Lock类中有一个指针成员,那么如果使用默认的析构函数、拷贝构造函数和赋值运算符,很可能会有严重的bug。

我们不妨在main函数中添加一句话,变成下面这样:

int main()
{
    int m = 5;
    Lock m1(&m);
    Lock m2(m1);
}

这里写图片描述
可见,锁被释放了两次,这就出问题了。原因是析构函数被调用了两次,在main()函数中生成了两个Lock对象,分别是m1和m2,Lock m2(m1)这句话使得m2.m_p = m1.m_p,这样这两个指针就指向了同一块资源。根据后生成的对象先析构的原则,所以m2先被析构,调用他的析构函数,释放资源锁,但释放的消息并没有通知到m1,所以m1在随后析构函数中,也会释放资源锁。

如果这里的释放不是简单的一句输出,而是真的对内存进行操作的话,程序就会崩溃。

归根到底,是程序使用了默认了拷贝构造函数造成的(当然,如果使用赋值运算的话,也会出现相同的bug),那么解决方案就是围绕如何正确摆平这个拷贝构造函数(和赋值运算符)。
第一个方法,很简单直观,就是干脆不让程序员使用类似于Lock m2(m1)这样的语句,一用就报编译错。这可以通过自己写一个私有的拷贝构造函数和赋值运算符的声明来解决。注意这里只要写声明就行了(见条款6)。

class Lock
{
public:
    explicit Lock(int* pm): m_p(pm)
    {
        lock(m_p);
    }

    ~Lock()
    {
        unlock(m_p);
    }

private:
    int *m_p;
    void lock(int* pm)
    {
        cout << "Address = " << pm << " is locked" << endl;
    }

    void unlock(int *pm)
    {
        cout << "Address = " << pm << " is unlocked" << endl;
    }

private:
    Lock(const Lock&);
    Lock& operator= (const Lock&);
};

这样编译就不会通过了:
当然也可以像书上写的一样,写一个Uncopyable的类,把它作为基类。在基类中把它的拷贝构造函数和赋值运算写成私有的(为了防止生成基类的对象,但又想允许派生类生成对象,可以把构造函数和析构函数的修饰符变成protected)。

然后
class Lock: public Uncopyable

{…}

也就OK了

第二个方法,就是使用shared_ptr来进行资源管理(见前一个条款),但还有一个问题,我想在生命周期结束后调用Unlock的方法,其实shared_ptr里面的删除器可以帮到我们。

只要这样子:

class Lock
{
public:
    explicit Lock(int *pm): m_p(pm, unlock){…}
private:
    shared_ptr<int> m_p;
}

这样在Lock的对象的生命周期结束后,就可以自动调用unlock了。

在条款十三的基础上,我改了一下自定义的shared_ptr,使之也支持删除器的操作了,代码如下:

#ifndef MY_SHARED_PTR_H
#define MY_SHARED_PTR_H

#include <iostream>
using namespace std;


typedef void (*FP)();    

template <class T>
class MySharedPtr
{

private:
    T *ptr;
    size_t *count;
    FP Del; // 声明一个删除器
    static void swap(MySharedPtr& obj1, MySharedPtr& obj2)
    {
        std::swap(obj1.ptr, obj2.ptr);
        std::swap(obj1.count, obj2.count);
        std::swap(obj1.Del, obj2.Del);
    }

public:
    MySharedPtr(T* p = NULL): ptr(p), count(new size_t(1)),Del(NULL){}

    // 添加带删除器的构造函数
    MySharedPtr(T* p, FP fun): ptr(p), count(new size_t(1)), Del(fun){}


    MySharedPtr(MySharedPtr& p): ptr(p.ptr), count(p.count), Del(p.Del)
    {
        ++ *p.count;
    }



    MySharedPtr& operator= (MySharedPtr& p)
    {
        if(this != &p && (*this).ptr != p.ptr)
        {
            MySharedPtr temp(p);
            swap(*this, temp);
        }
        return *this;
    }

    ~MySharedPtr()
    {
        if(Del != NULL)
        {
            Del();
        }    
        reset();
    }

    T& operator* () const
    {
        return *ptr;
    }

    T* operator-> () const 
    {
        return ptr;
    }

    T* get() const 
    {
        return ptr;
    }

    void reset()
    {
        -- *count;
        if(*count == 0)
        {
            delete ptr;
            ptr = 0;
            delete count;
            count = 0;
            //cout << "真正删除" << endl;
        }
    }

    bool unique() const
    {
        return *count == 1;
    }

    size_t use_count() const 
    {
        return *count;
    }


    friend ostream& operator<< (ostream& out, const MySharedPtr<T>& obj)
    {
        out << *obj.ptr;
        return out;
    }

};
#endif /* MY_SHARED_PTR_H */

带删除器的MySharedPtr

第三个方法是复制底部资源,就是将原来的浅拷贝转换成深拷贝,需要自己显示定义拷贝构造函数和赋值运算符。这个也在之前的条款说过了,放到这里,其实就是在拷贝的时候对锁的计数次数进行+1,析构函数里就是对锁的计数次数进行-1,如果减到0就去unlock(其实思想还是类似于shared_ptr进行资源管理)
第四个方法,是转移底部资源的控制权,这就是auto_ptr干的活了,在第二个方法中把shared_ptr换成auto_ptr就行了。

最后总结一下:

复制RAII对象必须一并复制它所管理的资源,所以资源copying行为决定RAII对象的copying行为
普遍而常见的RAII class copying行为是:抑制copying,施行引用计数法(shared_ptr思想),或者是转移底部资源(auto_ptr思想)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值