当前位置: 首页 > news >正文

c++11新特性——endable_shared_from_this

文章目录

  • 一.解决场景
    • 代码示例
    • 原因
  • 二.解决办法
    • 代码
  • 三.底层原理


一.解决场景

一个share_ptr管理的类,如果从类的函数里返回类对象(this指针),导致share_ptr引用计数错误,析构时异常问题

代码示例

#include <memory>
#include <iostream>class Bad
{
public:
std::shared_ptr<Bad> getptr() {
return std::shared_ptr<Bad>(this);
}
~Bad() { std::cout << "Bad::~Bad() called" << std::endl; }
};int main()
{
// 错误的示例,每个shared_ptr都认为自己是对象仅有的所有者
std::shared_ptr<Bad> bp1(new Bad());
std::shared_ptr<Bad> bp2 = bp1->getptr();
// 打印bp1和bp2的引用计数
std::cout << "bp1.use_count() = " << bp1.use_count() << std::endl;
std::cout << "bp2.use_count() = " << bp2.use_count() << std::endl;
} // Bad 对象将会被删除两次

在这里插入图片描述

原因

2个非共享的share_ptr指向同一个对象,未增加引用计数导对象被析构两次

二.解决办法

继承 std::enable_shared_from_this ,则会为该类 T 提供成员函数: shared_from_this 。 当 T 类型对象 t 被一个为名为 pt 的 std::shared_ptr 类对象管理时,调用 T::shared_from_this 成员函数,将会返回一个新的 std::shared_ptr 对象,它与 pt 共享 t 的所有权。

代码

#include <memory>
#include <iostream>struct Good : std::enable_shared_from_this<Good> // 注意:继承
{
public:
std::shared_ptr<Good> getptr() {
return shared_from_this();
}
~Good() { std::cout << "Good::~Good() called" << std::endl; }
};int main()
{
// 大括号用于限制作用域,这样智能指针就能在system("pause")之前析构
{
std::shared_ptr<Good> gp1(new Good());
std::shared_ptr<Good> gp2 = gp1->getptr();
// 打印gp1和gp2的引用计数
std::cout << "gp1.use_count() = " << gp1.use_count() << std::endl;
std::cout << "gp2.use_count() = " << gp2.use_count() << std::endl;
}
system("pause");
}

在这里插入图片描述

三.底层原理

通过继承enable_shared_from_this,然后调用share_from_this(),返回父类enable_shared_from_this的指针,然后shared_ptr在构造是检查是否可以将其转化来判断是否继承enable_shared_from_this

// 定义 enable_shared_from_this 模板类
template <typename T>
class enable_shared_from_this {
public:std::shared_ptr<T> shared_from_this() {return weak_this_.lock();  // 使用 weak_ptr 创建 shared_ptr}protected:// 构造函数,默认初始化 weak_this_enable_shared_from_this() {}private:std::weak_ptr<T> weak_this_;  // 用于存储对象的 weak_ptr// 允许 shared_ptr 访问私有成员template <typename U>friend class std::shared_ptr;
};// 自定义 shared_ptr 的构造函数
template <typename T>
class shared_ptr {
public:shared_ptr(T* ptr) : ptr_(ptr) {// 检查对象是否继承自 enable_shared_from_thisif (auto enable_shared = dynamic_cast<enable_shared_from_this<T>*>(ptr)) {enable_shared->weak_this_ = *this;  // 初始化 weak_this_}}// 其他 shared_ptr 成员函数...private:T* ptr_;  // 实际管理的对象指针
};// 示例类继承 enable_shared_from_this
class MyClass : public enable_shared_from_this<MyClass> {
public:void func() {std::shared_ptr<MyClass> self = shared_from_this();  // 获取指向自身的 shared_ptr// 使用 self 进行操作...}
};// 使用示例
int main() {std::shared_ptr<MyClass> obj = std::make_shared<MyClass>();obj->func();  // 正确使用 shared_from_this()return 0;
}
http://www.lryc.cn/news/436280.html

相关文章:

  • 小程序的右侧抽屉开关动画手写效果
  • vue3中el-table中点击图片放大时,被表格覆盖
  • GO学习笔记(4) strconv/time
  • 课程管理系统-数据库-基于MySQL的数据库课程设计
  • 降维打击 华为赢麻了
  • [数据集][目标检测]汽车头部尾部检测数据集VOC+YOLO格式5319张3类别
  • python 生成的代码,需要帮我生成一个直接在一台没有依赖的电脑上运行的 包
  • 【Linux】操作系统与进程
  • 【Linux】 LTG:移动硬盘部署Ubuntu24.04
  • Android的logcat日志详解
  • 【Linux】:信号的保存和信号处理
  • 深入理解Java虚拟机:Jvm总结-Java内存区域与内存溢出异常
  • 跨境电商必备保护账号的4个网络环境设置
  • Python+requests接口自动化测试框架实例教程
  • 【网络安全】DNS重绑定原理详析
  • C语言初识编译和链接
  • TrinityCore环境搭建
  • Proteus 仿真设计:开启电子工程创新之门
  • microchip dspic3一些奇怪问题
  • FinOps原则:云计算成本管理的关键
  • JavaScript之如何优化模板字符串的性能
  • 不能将类型“null”分配给类型“number | undefined”。ts(2322)
  • Nginx部署前端Vue项目详细教程
  • kvm 虚拟机命令行虚拟机操作、制作快照和恢复快照以及工作常用总结
  • 内网安全-横向移动【3】
  • 语言中的浮点数
  • Pyspark下操作dataframe方法(1)
  • 注解实现json序列化的时候自动进行数据脱敏
  • 使用Python下载文件的简易指南
  • 中秋国庆双节长假,景区迎来客流高峰,如何保障景区安全管理?