智能指针(二)

智能指针(二)

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

Android 的智能指针相关的源代码在如下两个文件中。

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

涉及的类及类之间的关系如图 6-4 所示。Android 中定义了两种智能指针类型,一种是强指针sp(strong pointer),一种是弱指针wp(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;

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值