Item 20: 使用std::weak_ptr替换会造成指针悬挂的类std::shared_ptr指针
本文翻译自modern effective C++,由于水平有限,故无法保证翻译完全正确,欢迎指出错误。谢谢!
博客已经迁移到这里啦
矛盾的是,我们很容易就能创造出一个和std::shared_ptr类似的智能指针,但是,它们不参加被指向资源的共享所有权管理。换句话说,这是一个行为像std::shared_ptr,但却不影响对象引用计数的指针。这样的智能指针需要与一个对std::shared_ptr来说不存在的问题做斗争:它指向的东西可能已经被销毁了。一个真正的智能指针需要通过追踪资源的悬挂(也就是说,被指向的对象不存在时)来解决这个问题。std::weak_ptr正好就是这种智能指针。
你可能会奇怪std::weak_ptr有什么用。当你检查std::weak_ptr的API时,你可能会更奇怪。它看起来一点也不智能。std::weak_ptr不能解引用,不能检查指针是否为空。这是因为std::weak_ptr不是独立的智能指针。它是std::shared_ptr的附加物。
它们的联系从出生起就存在了。std::weak_ptr常常创造自std::shared_ptr。std::shared_ptr初始化它们时,它们指向和std::shard_ptr指向的相同的位置,但是它们不影响它们所指向对象的引用计数:
auto spw = std::make_shared<Widget>(); //spw被构造之后,被指向的Widget
//的引用计数是1(关于std::make_shared
//的信息,看Item 21)
...
std::weak_ptr<Widget> wpw(spw); //wpw和spw指向相同的Widget,引用
//计数还是1
...
spw = nullptr; //引用计数变成0,并且Widget被销毁
//wpw现在是悬挂的
悬挂的std::weak_ptr被称为失效的(expired)。你能直接检查它:
if(wpw.expired())... //如果wpw不指向一个对象
但是为了访问std::weak_ptr指向的对象,你常常需要检查看这个std::weak_ptr是否已经失效了或者还没有失效(也就是,它没有悬挂)。想法总是比做起来简单,因为std::weak_ptr没有解引用操作,所以没办法写出相应的代码。即使能写出来,把解引用和检查分离开来会造成竞争条件:在调用expired和解引用操作中间,另外一个线程可能重新赋值或者销毁std::shared_ptr之前指向的对象,因此,会造成你想解引用的对象被销毁。这样的话,你的解引用操作将产生未定义行为。
你需要的是一个原子操作,它能检查看std::weak_ptr是否失效了,并让你能访问它指向的对象。从一个std::weak_ptr来创造std::shared_ptr就能达到这样的目的。你拥有的std::shared_ptr是什么样的,依赖于在你用std::weak_ptr来创建std::shared_ptr时是否已经失效了。操作有两种形式,一种是std::weak_ptr::lock,它返回一个std::shared_ptr。如果std::weak_ptr已经失效了,std::shared_ptr会是null:
std::shared_ptr<Widget> spw1 = wpw.lock(); //如果wpw已经失效了,spw1是null
auto spw2 = wpw.lock(); //和上面一样,不过用的是auto
另一种形式是参数为std::weak_ptr的std::shared_ptr的构造函数。这样情况下,如果std::weak_ptr已经失效了,会有一个异常抛出:
std::shared_ptr<Widget> spw3(wpw); //如果wpw已经失效了,抛出一个
//std::bad_weak_ptr异常
但是你可能还是对std::weak_ptr的用途感到奇怪。考虑一个工厂函数,这个函数根据唯一的ID,产生一个指向只读对象的智能指针。与Item 18的建议相符合,考虑工厂函数的返回类型,它返回一个std::unique_ptr:
std::unique_ptr<const Widget> loadWidget(WidgetId id);
如果loadWidget是一个昂贵的调用(比如,它执行文件操作或者I/O操作)并且对ID的反复使用是允许的,我们可以做一个合理的优化:写一个函数,这个函数做loadWidget做的事,但是它也缓存下它返回的结果。但是把所有请求的Widget都缓存下来会造成效率问题,所以另一个合理的优化是:当Widget不再使用时,销毁它的缓存。
对于这个缓存工厂函数,一个std::unique_ptr的返回类型是不够合适的。调用者应该收到一个指向缓存对象的智能指针,但是缓存也需要一个指针来指向对象。缓存的指针需要在他悬挂的时候能够察觉到,因为当工厂的客户把工厂返回的指针用完之后,对象将会被销毁,然后在缓存中相应的指针将会悬挂。因此缓存指针应该是一个std::weak_ptr(当指针悬挂的之后能够有所察觉)。这意味着工厂的返回值类型应该是一个std::shared_ptr,因为std::weak_ptr只有在对象的生命周期被std::shared_ptr管理的时候,才能检查自己是否悬挂。
这里给出一个缓存版本的loadWidget的快速实现:
std::shared_ptr<const Widget> fastLoadWidget(WidgetID id)
{
static std::unordered_map<WidgetID,
std::weak_ptr<const Widget> cache;
auto objPtr = cache[id].lock(); //objPtr是一个std::shared_ptr
/它指向缓存的对象(或者,当
//对象不在缓存中时为null)
if(!objPtr){ //吐过不在缓存中
objPtr = loadWidget(id); //加载它
cache[id] = objPtr; //缓存它
}
return objPtr;
}
这个实现使用了C++11的一种哈希容器(std::unordered_map),尽管它没有显示WidgetID的哈希函数以及比较函数,但他们还是会被实现出来的。
fastLoadWidget的实现忽略了一个事实,那就是缓存可能积累一些失效了的std::weak_ptr(对应的Widget已经不再被使用(因此这些Widget已经销毁了))。实现能被进一步优化,但是比起花费时间在这个问题(对std::weak_ptr的理解没有额外的提升)上,让我们考虑第二个使用场景:观察者设计模式。这个设计模式最重要的组件就是目标(subject,目标的状态可能会发生改变)和观察者(observer,当目标的状态发生改变时,观察者会被通知)。大多数实现中,每个目标包含一个数据成员,这个成员持有指向观察者的指针。这使得目标在状态发生改变的时候,通知起来更容易。目标对于控制他们的观察者的生命周期没有兴趣(也就是,当他们销毁时),但是它们对它们的观察者是否已经销毁了很有兴趣,这样它们就不会尝试去访问观察者了。一个合理的设计是:让每个目标持有一个容器,这个容器中装了指向它观察者的std::weak_ptr,因此,这让目标在使用一个指针前能确定它是否悬挂的。
最后一个std::weak_ptr的使用例子是:考虑一个关于A,B,C的数据结构,A和C共享B的所有权,因此都持有std::shared_ptr指向B:
假设从B指向A的指针同样有用,这个指针应该是什么类型的呢?
这里有三种选择:
一个原始指针。用这种方法,如果A销毁了,但是C仍然指向B,B将持有指向A的悬挂指针。B不会发现,所以B可能无意识地解引用这个悬挂指针。这将产生未定义的行为。
一个std::shared_ptr。在这种设计下,A和B互相持有指向对方的std::shared_ptr。这产生了std::shared_ptr的循环引用(A指向B,B指向A),这会阻止A和B被销毁。即使A和B无法从其他数据结构获得(比如,C不再指向B),A和B的引用计数都还是1.如果这发生了,A和B将被泄露,实际上:程序将不再能访问它们,这些资源也将不能被回收。
一个std::weak_ptr。这避免了上面的两个问题。如果A被销毁了,B中,指向A的指针将悬挂,但是B能察觉到。此外,尽管A和B会互相指向对方,B的指针也不会影响A的引用计数,因此A不再被指向时,B也不会阻止A被销毁。
使用std::weak_ptr是三个选择中最好的一个。但是,使用std::weak_ptr来预防std::shared_ptr的不常见的循环引用是不值得的。在严格分层的数据结构中,比如树,子节点通常只被它们的父节点拥有。当一个父节点被销毁时,它的子节点也应该被销毁。因此从父节点到子节点的连接通常被表示为std::unique_ptr。从子节点到父节点的链接能被安全地实现为原始指针,因为一个子节点的生命周期不应该比它们的父节点长。因此这里没有子节点对悬挂的父指针进行解引用的风险。
当然,不是所有基于指针的数据结构都是严格分层的,当这种情况发生时,就像上面的缓存和观察者链表的实现一样,我们知道std::weak_ptr已经跃跃欲试了。
从效率的观点来看,std::weak_ptr和std::shared_ptr在本质上是相同的。std::weak_ptr对象和std::shared_ptr一样大,它们和std::shared_ptr使用相同的控制块(看Item 19),并且构造,析构,赋值等操作也涉及到引用计数的原子操作。这可能会让你感到奇怪,因为我在这个Item的一开始就写了std::weak_ptr不参与引用计数的计算。我写的其实不是那个意思,我写的是,std::weak_ptr不参与共享对象的所有权,因此不会影响被指向对象的引用计数。控制块中其实还有第二个引用计数,这第二个引用计数是std::weak_ptr所维护的。细节部分,请继续看Item 21。
你要记住的事
- 使用std::weak_ptr替换那些会造成悬挂的类std::shared_ptr指针。
- 使用std::weak_ptr的潜在情况包括缓存,观察者链表,以及防止std::shared_ptr的循环引用。
Item 20: 使用std::weak_ptr替换会造成指针悬挂的类std::shared_ptr指针的更多相关文章
- std::shared_ptr 和 std::weak_ptr的用法以及引用计数的循环引用问题
在std::shared_ptr被引入之前,C++标准库中实现的用于管理资源的智能指针只有std::auto_ptr一个而已.std::auto_ptr的作用非常有限,因为它存在被管理资源的所有权转移 ...
- 智能指针std::weak_ptr
std::weak_ptr 避免shared_ptr内存泄漏的利器.
- std::weak_ptr
weak_ptr 是一种不控制对象生命周期的智能指针, 它指向一个 shared_ptr 管理的对象. 进行该对象的内存管理的是那个强引用的 shared_ptr. weak_ptr只是提供了对管理对 ...
- Qt 智能指针学习(7种QT智能指针和4种std智能指针)
从内存泄露开始? 很简单的入门程序,应该比较熟悉吧 ^_^ #include <QApplication> #include <QLabel> int main(int arg ...
- 标准C++类std::string的内存共享和Copy-On-Write...
标准C++类std::string的 内存共享和Copy-On-Write技术 陈皓 1. 概念 Scott Meyers在<More Effective C++>中举了个例子,不知你是否 ...
- 【转】标准C++类std::string的内存共享和Copy-On-Write技术
1. 概念 Scott Meyers在<More Effective C++>中举了个例子,不知你是否还记得?在你还在上学的时候,你的父母要你不要看电视,而去复习功 ...
- c++11 时间类 std::chrono
概念: chrono库:主要包含了三种类型:时间间隔Duration.时钟Clocks和时间点Time point. Duration:表示一段时间间隔,用来记录时间长度,可以表示几秒钟.几分钟或者几 ...
- 标准C++类std::string的内存共享和Copy-On-Write技术
标准C++类std::string的 内存共享和Copy-On-Write技术 陈皓 1. 概念 Scott Meyers在<More Effective C++>中举了个例子,不知你是 ...
- 智能指针 auto_ptr、scoped_ptr、shared_ptr、weak_ptr
什么是RAII? RAII是Resource Acquisition Is Initialization的简称,是C++语言的一种管理资源.避免泄漏的惯用法. RAII又叫做资源分配即初始化,即:定义 ...
随机推荐
- 记CSS格式化上下文
fomatting context 引言 主要讲解的是BFC上下文 本文是查看 史上最全面.最透彻的BFC原理剖析 的笔记 所以不会详解BFC, 只是记录学习心得, 以及重要规则避免原文失效 简介 F ...
- Nginx 安装与部署配置以及Nginx和uWSGI开机自启
下载 官方网站:https://nginx.org/en/download.html Windows下安装 安装 下载后解压(切记不能含有中文路径!!),文件结构如图(我解压的路径就有中文,记得拷贝放 ...
- OneAPM大讲堂 | 提高JavaScript性能的30个技巧
文章系国内领先的 ITOM 管理平台供应商 OneAPM 编译呈现. 您是网站管理员还是网页开发人员?想创建超快速的网站吗? 今天我们来看看 JavaScript,这项神奇而又复杂的技术.它使网站内容 ...
- 企业建立成功 DevOps 模式所需应对的5个挑战
[编者按]本文作者为 Kevin Goldberg,主要介绍要想成功部署 DevOps 模式,企业所需应对的5大挑战与问题.文章系国内 ITOM 管理平台 OneAPM 编译呈现. 要给 DevOps ...
- SQL Server Replication的分发服务器的快照文件夹位置查找
SQL Server分发服务器配置中,需要配置快照文件夹(Snapshot Folder),用于存储发布的数据和架构文件的工作目录,那么如何查找当前SQL Server数据库服务器的分发服务器的快照文 ...
- java导出数据到excel里:直接导出和导出数据库数据
一.直接导出 package com.ij34.util; import java.io.FileNotFoundException; import java.io.FileOutputStream; ...
- springmvc复习笔记----文件上传multipartResolver
结构 web.xml <?xml version="1.0" encoding=&q ...
- 使用SQL Developer生成Oracle数据库的关系模型(ER图)
客户要一张数据库的关系模型图,于是用SQL Developer来做. 一.SQL Developer版本 我在官网下载的最新版本(现在已经到了18.1,Oracle更新的太勤快): 2.如下图所示选择 ...
- shell脚本:通过域名获取证书的过期时间
需要两个文件,一个用于存储域名信息,另一个是检测脚本 注意:这两个文件是在一个目录下 domain_ssl.info [存储域名信息] [root@mini05 ]# cat domain_ssl.i ...
- Servlet(四):转发与重定向、路径问题
在上次的小案例中用到了转发的技术,今天来仔细聊聊转发和重定向的问题,以及一些小知识的汇总. 一.转发 1.转发的概念 转发主要是将浏览器的请求交给另外一个servlet或jsp来处理,借助reques ...