杂学杂记(六)智能指针简介(强指针、弱指针)

智能指针的原理及实现

       当类中有 指针成员时,一般有两种方式来管理 指针成员:一是采用值型的方式管理,每个类对象都保留一份指针指向的对象的拷贝;另一种更优雅的方式是使用智能指针,从而实现指针指向的对象的共享。
智能 指针(smart pointer)的一种通用实现技术是使用 引用计数(reference count)。智能 指针类将一个计数器与类指向的对象相关联, 引用计数跟踪该类有多少个对象共享同一指针。
       每次创建类的新对象时,初始化指针并将 引用计数置为1;当对象作为另一对象的父本而创建时, 拷贝构造函数拷贝指针并增加与之相应的引用计数;对一个对象进行赋值时,赋值操作符减少左 操作数所指对象的引用计数(如果引用计数为减至0,则删除对象),并增加右操作数所指对象的引用计数;调用 析构函数时,析构函数减少引用计数(如果引用计数减至0,则删除基础对象)。
   

       在C++和Android的源代码中,经常会看到形如:sp<xxx>、wp<xxx>这样的类型定义,这其实就是智能指针。智能指针是C++中的一个概念,通过基于引用计数的方法,解决对象的自动释放的问题。在C++编程中,有两个很让人头痛的问题:一是忘记释放动态申请的对象从而造成内存泄露;二是对象在一个地方释放后,又在别的地方被使用,从而引起内存访问错误。程序员往往需要花费很大精力进行精心设计,以避免这些问题的出现。在使用智能指针后,动态申请的内存将会被自动释放(有点类似Java的垃圾回收),不需要再使用delete来释放对象,也不需要考虑一个对象是否已经在其它地方被释放了,从而使程序编写工作减轻不少,而程序的稳定性大大提高。

     boost库里就有定义了智能指针,使用boost::shared_ptr<类名> 指针名来使用智能指针,比如:boost::shared_ptr<SnapCatch> SnapCatch_ptr; 而在Android中,智能指针相关的源代码在下面两个文件中:

    frameworks/base/include/utils/RefBase.h
    frameworks/base/libs/utils/RefBase.cpp

使用sp<类名> 指针名来使用,比如sp<SnapCatch> SnapCatch_ptr;

    智能指针包括两种类型,一种是强指针sp(strong pointer),一种是弱指针(weak pointer)。其实成为强引用和弱引用更合适一些。强指针与一般意义的智能指针概念相同,通过引用计数来记录有多少使用者在使用一个对象,如果所有使用者都放弃了对该对象的引用,则该对象将被自动销毁。

    弱指针也指向一个对象,但是弱指针仅仅记录该对象的地址,不能通过弱指针来访问该对象,也就是说不能通过弱智真来调用对象的成员函数或访问对象的成员变量。要想访问弱指针所指向的对象,需首先将弱指针升级为强指针(通过wp类所提供的promote()方法)。弱指针所指向的对象是有可能在其它地方被销毁的,如果对象已经被销毁,wp的promote()方法将返回空指针,这样就能避免出现地址访问错的情况。

    是不是很神奇?弱指针是怎么做到这一点的呢?其实说穿了一点也不复杂,原因就在于每一个可以被智能指针引用的对象都同时被附加了另外一个weakref_impl类型的对象,这个对象中负责记录对象的强指针引用计数和弱指针引用计数。这个对象是智能指针的实现内部使用的,智能指针的使用者看不到这个对象。弱指针操作的就是这个对象,只有当强引用计数和弱引用计数都为0时,这个对象才会被销毁。

    说了这么多原理,下面该看看到底智能指针该怎么使用了。假设现在有一个类MyClass,如果要使用智能指针来引用这个类的对象,那么这个类需满足下列两个前提条件:

    (1) 这个类是基类RefBase的子类或间接子类;
    (2) 这个类必须定义虚构造函数,即它的构造函数需要这样定义:

          virtual ~MyClass();

    满足了上述条件的类就可以定义智能指针了,定义方法和普通指针类似。比如普通指针是这样定义:

    MyClass* p_obj;

    智能指针是这样定义:

    sp<MyClass> p_obj;

    注意不要定义成 sp<MyClass>* p_obj。初学者容易犯这种错误,这样实际上相当于定义了一个指针的指针。尽管在语法上没有问题,但是最好永远不要使用这样的定义。

    定义了一个智能指针的变量,就可以象普通指针那样使用它,包括赋值、访问对象成员、作为函数的返回值、作为函数的参数等。比如:

    p_obj = new MyClass(); // 注意不要写成 p_obj = new sp<MyClass>

    sp<MyClass> p_obj2 = p_obj;

    p_obj->func();

    p_obj = create_obj();

    some_func(p_obj);

    注意不要试图delete一个智能指针,即 delete p_obj。不要担心对象的销毁问题,智能指针的最大作用就是自动销毁不再使用的对象。不需要再使用一个对象后,直接将指针赋值为NULL即可:

    p_obj = NULL;

    上面说的都是强指针,弱指针的定义方法和强指针类似,但是不能通过弱指针来访问对象的成员。下面是弱指针的示例:

    wp<MyClass> wp_obj = new MyClass();

    p_obj = wp_obj.promote(); // 升级为强指针。不过这里要用.而不是->,真是有负其指针之名啊

    wp_obj = NULL;

   

    智能指针用起来是很方便,在一般情况下最好使用智能指针来代替普通指针。但是需要知道一个智能指针其实是一个对象,而不是一个真正的指针,因此其运行效率是远远比不上普通指针的。所以在对运行效率敏感的地方,最好还是不要使用智能指针为好。

你可能感兴趣的:(杂学杂记(六)智能指针简介(强指针、弱指针))