智能指针 shared_ptr weak_ptr shared_from_this 笔记
shared_ptr
当指向对象的std::shared_ptr一创建,被管理对象的控制块SharedPtrControlBlock(参考下面的图)就建立了。
被管理的对象的控制块中有引用计数(reference count),当引用计数为0时,被管理的对象会被销毁。
控制块的创建会遵循下面几条规则:
std::make_shared会创建控制块
从 std::unique_ptr 上构造出 std::shared_ptr的时候,会创建控制块
从原始指针上构造 std::shared_ptr的时候会创建控制块
假若我们为一个指针构建了多个控制块,那么会有多个引用计数,意味着对象可能销毁多次,这是非法行为
多次从原始指针上构建智能指针,实际存在多个互不相干的引用计数,大家都为1,然后一旦有一个智能指针不再指向该对象,引用为0就会析构对象,就会出错
定义智能指针的方式
std::shared_ptr<A> p1(new A);
std::shared_ptr<A> p2 = std::make_shared<A>();
在执行std::shared_ptrp1(new A) 的时候,首先会申请数据的内存,然后申请内控制块,因此是两次内存申请
而std::make_shared()则是只执行一次内存申请,将数据和控制块的申请放到一起
图片来自
std::enable_shared_from_this 有什么意义? - 孔洽孔洽的回答 - 知乎https://www.zhihu.com/question/30957800/answer/2700292012
weak_ptr
还记得,shared_ptr内含有指向计数区域(SharedPtrControlBlock)结构体的指针吧
struct SharedPtrControlBlock{
int shared_count;
int weak_count;
};
该结构体引进新的int变量weak_count,来作为弱引用计数
每个weak_ptr都占指针的两倍空间,一个装着原始指针,一个装着计数区域的指针(和shared_ptr一样的成员)
大概是这样
class weak_ptr{
T* ptr;
SharedPtrControlBlock* count;
};
被管理资源的释放只取决于shared计数,当shared计数为0,才会释放被管理资源,
但是控制块计数区域的释放却取决于shared计数和weak计数,当两者均为0时,才会释放计数区域
这也就是weak_ptr能够使用expired方法判断对象是否被析构的原因,因为该计数区域还在
weak_ptr 一般者是通过 shared _ptr 或另一个weak_ptr 来初始化
weak_ptr不会改变资源的引用计数,只是一个观察者的角色,通过观察shared_ptr来判定资源是否存在,不会影响指向对象(被管理资源)的生命周期,而shared_ptr引用计数为0则会析构对象
weak_ptr持有的引用计数,不是资源的引用计数,而是同一个资源的观察者的计数
weak_ptr没有提供常用的指针操作,无法直接访问资源,需要先通过lock方法提升为shared_ptr强智能指针,才能访问资源
成员函数use_count() 观测资源引用计数
成员函数expired() 功能相当于 use_count()为 0 表示被观测的资源(也就是shared_ptr的管理的资源)被销毁
成员函数lock()从被观测的shared_ptr获得一个可用的shared_ptr对象, 进而操作资源。但当expired()==true的时候,lock()函数将返回一个存储空指针的shared_ptr
来自csdn
原文链接:https://blog.csdn.net/qq_53111905/article/details/122240842
循环引用问题
看下面一段代码
weak_ptr用于解决两个类内部使用shared_ptr互相引用造成的循环,TestA和TestB中weak_ptr如果换成是shared_ptr
那么在离开main函数时,两个在main中创建的shared_ptr失效,本来引用数都是2,各自减一都变为了1,也就是内部的shared_ptr互相指向,成为一个环
两个对象都还被引用着,所以不会析构,造成内存泄露
所以将shared_ptr都改为weak_ptr,离开main时两个在main中创建的shared_ptr都会失效,各自的引用减一,变为0,内部的弱指针不影响引用计数,故而能够正确释放内存
//示例代码1.0 http://www.cnblogs.com/feng-sc/p/5710724.html
#include <memory>
class TestB;
class TestA
{
public:
TestA()
{
std::cout << "TestA()" << std::endl;
}
void ReferTestB(std::shared_ptr<TestB> test_ptr)
{
m_TestB_Ptr = test_ptr;
}
void TestWork()
{
std::cout << "~TestA::TestWork()" << std::endl;
}
~TestA()
{
std::cout << "~TestA()" << std::endl;
}
private:
std::weak_ptr<TestB> m_TestB_Ptr;
};
class TestB
{
public:
TestB()
{
std::cout << "TestB()" << std::endl;
}
void ReferTestB(std::shared_ptr<TestA> test_ptr)
{
m_TestA_Ptr = test_ptr;
}
void TestWork()
{
std::cout << "~TestB::TestWork()" << std::endl;
}
~TestB()
{
std::shared_ptr<TestA> tmp = m_TestA_Ptr.lock();
tmp->TestWork();
std::cout << "2 ref a:" << tmp.use_count() << std::endl;
std::cout << "~TestB()" << std::endl;
}
std::weak_ptr<TestA> m_TestA_Ptr;
};
int main()
{
std::shared_ptr<TestA> ptr_a = std::make_shared<TestA>();
std::shared_ptr<TestB> ptr_b = std::make_shared<TestB>();
ptr_a->ReferTestB(ptr_b);
ptr_b->ReferTestB(ptr_a);
std::cout << "1 ref a:" << ptr_a.use_count() << std::endl;
std::cout << "1 ref b:" << ptr_a.use_count() << std::endl;
return 0;
}
shared_from_this要解决的问题
需求: 在类的内部需要自身的shared_ptr 而不是this裸指针,直接从this指针创建?那会导致出现多个引用计数器,会错误析构,因为每调用一次就创建一个指针
场景: 在类中发起一个异步操作, callback回来要保证发起操作的对象仍然有效,std::bind函数和对象,结果对象先被析构了,那还调用个屁
异步回调的时候对象可能已经被销毁了 所以使用shared_ptr 传出去就是保证最少还有一个引用计数维持对象的生命周期直到回调结束
struct A {
void func() {
std::shared_ptr<A> local_sp_a(this);
// do something with local_sp_a
}
};
上面这段代码,由成员函数直接在this上构建智能指针,当离开该函数作用域,local_sp_a就会失效,则引用计数少1,则会立即析构this
改成这样
struct A : public enable_shared_from_this {
void func() {
std::shared_ptr<A> local_sp_a = shared_from_this();
// do something with local_sp
}
};
shared_from_this()会查找当前对象控制块,然后创建一个新的std::shared_ptr关联这个控制块。
用这个函数之前,是假设当前对象已经存在一个关联的控制块。因此,必须已经存在一个指向当前对象的std::shared_ptr。
如果没有std::shared_ptr指向当前对象(即当前对象没有关联控制块),那么shared_from_this会抛出一个异常。
智能指针出现的意义
智能指针的应用场景其实大概就了解了,
有时,程序不能正常释放内存资源,忘记释放就不说了,如出现异常,后面的代码不会执行,可能析构对象的代码就被忽略了
有时候,对象已经不存在了,被析构了,然而程序仍在调用该对象的成员函数,则会出错,可以通过weak_ptr判断对象是否还在
异步,以及多线程的情况下,都会出现对象可能不存在了的情况
可以通过weak_ptr的lock方法判断是否为nullptr来判断对象是否被析构
智能指针使用总结
当你需要一个独占资源所有权(访问权+生命控制权)的指针,且不允许任何外界访问,使用std::unique_ptr
当你需要一个共享资源所有权(访问权+生命控制权)的指针,使用std::shared_ptr
当你需要一个能访问资源,但不控制其生命周期的指针,使用std::weak_ptr
一个shared_ptr和n个weak_ptr搭配使用而不是n个shared_ptr
因为一般模型中,最好总是被一个指针控制生命周期,然后可以被n个指针控制访问
参考链接:https://www.cnblogs.com/KillerAery/p/9096558.html
智能指针 shared_ptr weak_ptr shared_from_this 笔记的更多相关文章
- 深入学习c++--智能指针(二) weak_ptr(打破shared_ptr循环引用)
1. 几种智能指针 1. auto_ptr: c++11中推荐不使用他(放弃) 2. shared_ptr: 拥有共享对象所有权语义的智能指针 3. unique_ptr: 拥有独有对象所有权语义的智 ...
- [6] 智能指针boost::weak_ptr
[1]boost::weak_ptr简介 boost::weak_ptr属于boost库,定义在namespace boost中,包含头文件 #include<boost/weak_ptr.hp ...
- c/c++ 智能指针 shared_ptr 使用
智能指针 shared_ptr 使用 上一篇智能指针是啥玩意,介绍了什么是智能指针. 这一篇简单说说如何使用智能指针. 一,智能指针分3类:今天只唠唠shared_ptr shared_ptr uni ...
- C++智能指针shared_ptr
shared_ptr 这里有一个你在标准库中找不到的—引用数智能指针.大部分人都应当有过使用智能指针的经历,并且已经有很多关于引用数的文章.最重要的一个细节是引用数是如何被执行的—插入,意思是说你将引 ...
- STL源码剖析-智能指针shared_ptr源码
目录一. 引言二. 代码实现 2.1 模拟实现shared_ptr2.2 测试用例三. 潜在问题分析 你可能还需要了解模拟实现C++标准库中的auto_ptr一. 引言与auto_ptr大同小异,sh ...
- c/c++ 智能指针 shared_ptr 和 new结合使用
智能指针 shared_ptr 和 new结合使用 用make_shared函数初始化shared_ptr是最推荐的,但有的时候还是需要用new关键字来初始化shared_ptr. 一,先来个表格,唠 ...
- 智能指针shared_ptr新特性shared_from_this及weak_ptr
enable_shared_from_this是一个模板类,定义于头文件<memory>,其原型为: template< class T > class enable_shar ...
- 智能指针shared_ptr的用法
为了解决C++内存泄漏的问题,C++11引入了智能指针(Smart Pointer). 智能指针的原理是,接受一个申请好的内存地址,构造一个保存在栈上的智能指针对象,当程序退出栈的作用域范围后,由于栈 ...
- 智能指针之 weak_ptr
1. weak_ptr 介绍 std::weak_ptr 是一种智能指针,它对被 std::shared_ptr 管理的对象存在非拥有性("弱")引用.在访问所引用的对象指针前必须 ...
- 智能指针 shared_ptr 解析
近期正在进行<Effective C++>的第二遍阅读,书里面多个条款涉及到了shared_ptr智能指针,介绍的太分散,学习起来麻烦.写篇blog整理一下. LinJM @HQU s ...
随机推荐
- 动态SQL遇到的问题
看图 查不出来任何数据 因为判断有问题 修改方法如下:
- ArcObjects SDK开发 014 MapSurround和普通Element
1.如何获取MapSurround 和获取MapFrame类似,如果你已经获取指北针.比例尺等对象,可以通过IGraphicsContainer的FindFrame函数获取.如果没有,则通过IGrap ...
- C#实现文件导入与导出
无论是文件的导入与导出都需要引入IO库,引入方法如下: using System.IO; 通过以下代码可以实现将文件导入到数组中 string path;//定义一个路径 OpenFileDialog ...
- 或许是市面上最强的 Mock 工具
背景 在开发过程中,由于后端与前端并行开发,或者前端需要等待后台开发,难以保证对接效率,同时即使用开发好的 API 对接,也有可能一个 API 不通就阻塞了整个软件的对接工作.同时对软件的敏感度也很高 ...
- $_GET方法踩坑
背景 写代码时,遇到一个奇怪的问题:小程序卡券解码接口老是报解码失败,苦寻了一个小时,发现原来是url参数经过thinkphp的I方法被过滤掉,而且涉及到PHP原生的$_GET 原因 I方法底层是原生 ...
- DSS+Linkis Ansible 单机一键安装脚本
DSS+Linkis Ansible 单机一键安装脚本 一.简介 为解决繁琐的部署流程,简化安装步骤,本脚本提供一键安装最新版本的DSS+Linkis环境:部署包中的软件采用我自己编译的安装包,并且为 ...
- 【转载】EXCEL VBA 选取非连续的单元格区域——Areas集合
出处:http://www.360doc.com/content/21/1113/17/77710807_1004011085.shtml 前面我们讲的大多是**并操作单个的单元格,或者是连续的单元格 ...
- CLISP学习(一)
Lisp Lisp发表于1960,是由数学家约翰·麦卡锡发明的. 在计算机有所发展时,数学家们对人工智能有着极大的兴趣,觉得可以实现一个共通的方法,使计算机能够处理链表中的符号数据,允许语言的处理.信 ...
- VSCode运行C/C++配置
将MinGw安装目录下的 1.安装 VSCode 2.安装 MinGW 链接:点击跳转 3.MinGW 内安装两个模块 1.右键 Mark for Installation 勾选 (此处已安装好,所以 ...
- 关系数据库——MySQL
数据库 1.基本操作 1.1命令行操作 mysql -u username -p+password; --连接数据库 flush privileges; --刷新权限 show databases; ...