智能指针(weak_ptr )之三

1. std::weak_ptr

1.1 定义与用法

std::weak_ptr 是一种不拥有对象所有权的智能指针,用于观察但不影响对象的生命周期。主要用于解决 shared_ptr 之间的循环引用问题。

主要特性

  • 非拥有所有权:不增加引用计数。
  • 可从 shared_ptr 生成:通过 std::weak_ptr 可以访问 shared_ptr 管理的对象。
  • 避免循环引用:适用于双向关联或观察者模式。

1.2 避免循环引用

在存在双向关联(如父子关系)时,使用多个 shared_ptr 可能导致循环引用,导致内存泄漏。此时,可以使用 weak_ptr 来打破循环。

1.3 代码案例

场景:带有循环引用的观察者模式(天气预报系统)
#include <iostream>
#include <vector>
#include <memory>
#include <algorithm>// -------------------- 抽象接口部分 --------------------// 观察者接口:所有 App 都要实现这个接口
class Observer {
public:virtual void update(float temp) = 0; // 被通知时调用virtual ~Observer() {std::cout << "Observer destroyed" << std::endl;}
};// 主题(被观察者)接口:天气预报系统实现它
class Subject {
public:virtual void attach(std::shared_ptr<Observer> observer) = 0; // 添加观察者virtual void detach(std::shared_ptr<Observer> observer) = 0; // 移除观察者virtual void notify() = 0;                                   // 通知所有观察者virtual ~Subject() {std::cout << "Subject destroyed" << std::endl;}
};// -------------------- 主题实现部分 --------------------// 天气预报系统,持有观察者的 shared_ptr 列表
class WeatherStation : public Subject {
private:std::vector<std::shared_ptr<Observer>> observers; // 所有订阅者float temperature = 0.0f;public:// 注册观察者void attach(std::shared_ptr<Observer> observer) override {observers.push_back(observer);}// 注销观察者void detach(std::shared_ptr<Observer> observer) override {observers.erase(std::remove(observers.begin(), observers.end(), observer), observers.end());}// 通知所有观察者,调用它们的 update()void notify() override {for (auto& obs : observers) {obs->update(temperature);}}// 模拟温度变化,触发通知void setTemperature(float t) {temperature = t;std::cout << "[WeatherStation] New temperature: " << t << "°C\n";notify();}~WeatherStation() {std::cout << "[WeatherStation] destroyed\n";}
};// -------------------- 观察者实现部分 --------------------// App 实现观察者接口:当温度太高,会自动注销
class AppClient : public Observer, public std::enable_shared_from_this<AppClient> {
private:std::shared_ptr<WeatherStation> station; // ❌ 持有 shared_ptr,形成循环引用public:// 构造函数,记录主题指针AppClient(std::shared_ptr<WeatherStation> s) : station(s) {}// 被通知时调用void update(float temp) override {std::cout << "[AppClient] Received temperature: " << temp << "°C\n";// 如果温度过高,就自动注销if (temp > 35.0f) {std::cout << "[AppClient] Too hot! Unsubscribing...\n";// ❌ 使用 shared_from_this() 注销自己,导致循环引用无法释放station->detach(shared_from_this());}}// 析构函数~AppClient() {std::cout << "[AppClient] destroyed\n";}
};// -------------------- 主函数 --------------------int main() {// 创建主题对象(天气系统)std::shared_ptr<WeatherStation> station = std::make_shared<WeatherStation>();// 创建观察者对象(App),并订阅std::shared_ptr<AppClient> app = std::make_shared<AppClient>(station);station->attach(app);// 第一次更新温度,通知观察者station->setTemperature(32.0f);// 第二次更新温度,观察者会自动注销// station->setTemperature(36.5f);  //如果解除这一步,那么不会产生引用循环// 程序即将结束std::cout << "[main] Exiting...\n";return 0;
}

输出

结果:产生引用循环Observer 和 Subject  没有销毁 导致内存泄漏[WeatherStation] New temperature: 32°C
[AppClient] Received temperature: 32°C
[main] Exiting...----------------------------------------------------------------------------//解除station->setTemperature(36.5f);				
结果:	不会产生引用循环[WeatherStation] New temperature: 32°C  
[AppClient] Received temperature: 32°C  
[WeatherStation] New temperature: 36.5°C
[AppClient] Received temperature: 36.5°C
[AppClient] Too hot! Unsubscribing...   
[main] Exiting...
[AppClient] destroyed
Observer destroyed
[WeatherStation] destroyed
Subject destroyed
解决方案:使用 weak_ptr

改用 weak_ptr 其中一方,打破循环引用。

方式一 把AppClient里面的station改为weak_ptr类型,其他操作也做出相应改变

//方式一 把AppClient里面的station改为weak_ptr类型,其他操作也做出相应改变
#include <iostream>
#include <vector>
#include <memory>
#include <algorithm>// -------------------- 抽象接口部分 --------------------// 观察者接口:所有 App 都要实现这个接口
class Observer {
public:virtual void update(float temp) = 0; // 被通知时调用virtual ~Observer() {std::cout << "Observer destroyed" << std::endl;}
};// 主题(被观察者)接口:天气预报系统实现它
class Subject {
public:virtual void attach(std::shared_ptr<Observer> observer) = 0; // 添加观察者virtual void detach(std::shared_ptr<Observer> observer) = 0; // 移除观察者virtual void notify() = 0;                                   // 通知所有观察者virtual ~Subject() {std::cout << "Subject destroyed" << std::endl;}
};// -------------------- 主题实现部分 --------------------// 天气预报系统,持有观察者的 shared_ptr 列表
class WeatherStation : public Subject {
private:std::vector<std::shared_ptr<Observer>> observers; // 所有订阅者float temperature = 0.0f;public:// 注册观察者void attach(std::shared_ptr<Observer> observer) override {observers.push_back(observer);}// 注销观察者void detach(std::shared_ptr<Observer> observer) override {observers.erase(std::remove(observers.begin(), observers.end(), observer), observers.end());}// 通知所有观察者,调用它们的 update()void notify() override {for (auto& obs : observers) {obs->update(temperature);}}// 模拟温度变化,触发通知void setTemperature(float t) {temperature = t;std::cout << "[WeatherStation] New temperature: " << t << "°C\n";notify();}~WeatherStation() {std::cout << "[WeatherStation] destroyed\n";}
};// -------------------- 观察者实现部分 --------------------// App 实现观察者接口:当温度太高,会自动注销
class AppClient : public Observer, public std::enable_shared_from_this<AppClient> {
private:std::weak_ptr<WeatherStation> station;  //修改1public:// 构造函数,记录主题指针AppClient(std::shared_ptr<WeatherStation> s) : station(s) {}// 被通知时调用void update(float temp) override {std::cout << "[AppClient] Received temperature: " << temp << "°C\n";// 如果温度过高,就自动注销if (temp > 35.0f) {std::cout << "[AppClient] Too hot! Unsubscribing...\n";if(!station.expired()) //修改2{station.lock()->detach(shared_from_this()); }}}// 析构函数~AppClient() {std::cout << "[AppClient] destroyed\n";}
};// -------------------- 主函数 --------------------int main() {// 创建主题对象(天气系统)std::shared_ptr<WeatherStation> station = std::make_shared<WeatherStation>();// 创建观察者对象(App),并订阅std::shared_ptr<AppClient> app = std::make_shared<AppClient>(station);station->attach(app);// 第一次更新温度,通知观察者station->setTemperature(32.0f);// 第二次更新温度,观察者会自动注销// station->setTemperature(36.5f);  //如果解除这一步,那么不会产生引用循环// 程序即将结束std::cout << "[main] Exiting...\n";return 0;
}

输出

[WeatherStation] New temperature: 32°C
[AppClient] Received temperature: 32°C
[main] Exiting...
[WeatherStation] destroyed
[AppClient] destroyed
Observer destroyed
Subject destroyed

方式二 把WeatherStation里面的observers改为weak_ptr类型,其他操作也做出相应改变

#include <iostream>
#include <vector>
#include <memory>
#include <algorithm>// -------------------- 抽象接口部分 --------------------// 观察者接口:所有 App 都要实现这个接口
class Observer
{
public:virtual void update(float temp) = 0; // 被通知时调用virtual ~Observer(){std::cout << "Observer destroyed" << std::endl;}
};// 主题(被观察者)接口:天气预报系统实现它
class Subject
{
public:virtual void attach(std::shared_ptr<Observer> observer) = 0; // 添加观察者virtual void detach(std::shared_ptr<Observer> observer) = 0; // 移除观察者virtual void notify() = 0;                                   // 通知所有观察者virtual ~Subject(){std::cout << "Subject destroyed" << std::endl;}
};// -------------------- 主题实现部分 --------------------// 天气预报系统,持有观察者的 shared_ptr 列表
class WeatherStation : public Subject
{
private:std::vector<std::weak_ptr<Observer>> observers; // 修改1float temperature = 0.0f;public:// 注册观察者void attach(std::shared_ptr<Observer> observer) override{observers.push_back(observer);}// 注销观察者void detach(std::shared_ptr<Observer> observer) override{observers.erase(    // 修改2 std::remove_if(observers.begin(), observers.end(),[&observer](const std::weak_ptr<Observer> &wptr){return !wptr.expired() && wptr.lock() == observer;}),observers.end());}// void detach(std::shared_ptr<Observer> observer) override {//        std::cout << "[WeatherStation] detach\n";//        std::weak_ptr<Observer> temp = observer;//        observers.erase(std::remove(observers.begin(), observers.end(), temp), observers.end());//   }//如果改成这样的话,思路很接近核心,看起来像是合理的,但它其实 仍然不行//原因就在于:std::weak_ptr 不支持 == 比较(哪怕是彼此之间),除非你先把它们 lock 成 shared_ptr//两个 weak_ptr 即使指向相同的对象,它们也可能过期(expired)//无法确定“空的”和“已销毁”的哪个该算“相等”// 通知所有观察者,调用它们的 update()void notify() override{for (auto &obs : observers){if (!obs.expired()) // 修改3obs.lock()->update(temperature); }}// 模拟温度变化,触发通知void setTemperature(float t){temperature = t;std::cout << "[WeatherStation] New temperature: " << t << "°C\n";notify();}~WeatherStation(){std::cout << "[WeatherStation] destroyed\n";}
};// -------------------- 观察者实现部分 --------------------// App 实现观察者接口:当温度太高,会自动注销
class AppClient : public Observer, public std::enable_shared_from_this<AppClient>
{
private:std::shared_ptr<WeatherStation> station; // ❌ 持有 shared_ptr,形成循环引用public:// 构造函数,记录主题指针AppClient(std::shared_ptr<WeatherStation> s) : station(s) {}// 被通知时调用void update(float temp) override{std::cout << "[AppClient] Received temperature: " << temp << "°C\n";// 如果温度过高,就自动注销if (temp > 35.0f){std::cout << "[AppClient] Too hot! Unsubscribing...\n";// ❌ 使用 shared_from_this() 注销自己,导致循环引用无法释放station->detach(shared_from_this());}}// 析构函数~AppClient(){std::cout << "[AppClient] destroyed\n";}
};// -------------------- 主函数 --------------------int main()
{// 创建主题对象(天气系统)std::shared_ptr<WeatherStation> station = std::make_shared<WeatherStation>();// 创建观察者对象(App),并订阅std::shared_ptr<AppClient> app = std::make_shared<AppClient>(station);station->attach(app);// 第一次更新温度,通知观察者station->setTemperature(32.0f);// 第二次更新温度,观察者会自动注销// station->setTemperature(36.5f);  //如果解除这一步,那么不会产生引用循环// 程序即将结束std::cout << "[main] Exiting...\n";return 0;
}

输出

[WeatherStation] New temperature: 32°C
[AppClient] Received temperature: 32°C
[main] Exiting...
[AppClient] destroyed
Observer destroyed
[WeatherStation] destroyed
Subject destroyed

1.4 访问 weak_ptr 指向的对象

weak_ptr 不能直接访问对象,需要通过 lock() 方法转换为 shared_ptr,并检查对象是否仍然存在。

  • std::weak_ptr::expired()
    • 功能:std::weak_ptr 的 expired() 成员函数用于检查 std::weak_ptr 所引用的对象是否已经被销毁(即关联的 std::shared_ptr 的引用计数是否已变为 0)。
    • 返回值:
      • bool类型
        • true:表示 std::weak_ptr 已过期(引用的对象已被销毁)。
        • false:表示 std::weak_ptr 未过期(引用的对象仍然存在)。
#include <iostream>
#include <memory>int main() {std::shared_ptr<int> sp = std::make_shared<int>(42);std::weak_ptr<int> wp = sp;if (auto locked = wp.lock()) { //wp.lock() 会尝试安全地获取一个新的 shared_ptr,如果资源还存在,它就返回一个有效的 								 //shared_ptr,否则返回一个空的 shared_ptr 到时if(shared_ptr)会触发operator bool()函								 //	数,也就是if(shared_ptr.bool()) == if(return shared_ptr.get() != nullptr;)std::cout << "Value: " << *locked << std::endl;} else {std::cout << "Object no longer exists." << std::endl;}sp.reset(); // 释放资源if (auto locked = wp.lock()) { // 再次尝试获取 shared_ptr std::cout << "Value: " << *locked << std::endl;} else {std::cout << "Object no longer exists." << std::endl;}/*	效果一样!expired()if(!wp.expired()) { // 再次尝试获取 shared_ptr std::cout << "Value: " << *wp.lock() << std::endl;} else {std::cout << "Object no longer exists." << std::endl;}*/return 0;
}

输出

Value: 42
Object no longer exists.

解析

  • wp.lock() 返回一个 shared_ptr,如果对象依然存在,则有效。
  • sp.reset() 释放资源后,wp.lock() 无法获取有效的 shared_ptr

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/77871.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

学习海康VisionMaster之卡尺工具

一&#xff1a;进一步学习了 今天学习下VisionMaster中的卡尺工具&#xff1a;主要用于测量物体的宽度、边缘的特征的位置以及图像中边缘对的位置和间距 二&#xff1a;开始学习 1&#xff1a;什么是卡尺工具&#xff1f; 如果我需要检测芯片的每一个PIN的宽度和坐标&#xff…

Java面试实战:从Spring Boot到微服务的深入探讨

Java面试实战&#xff1a;从Spring Boot到微服务的深入探讨 场景&#xff1a;电商场景的面试之旅 在某互联网大厂的面试间&#xff0c;面试官李老师正襟危坐&#xff0c;而对面坐着的是传说中的“水货程序员”赵大宝。 第一轮&#xff1a;核心Java与构建工具 面试官&#x…

深入理解 Spring @Configuration 注解

在 Spring 框架中,@Configuration 注解是一个非常重要的工具,它用于定义配置类,这些类可以包含 Bean 定义方法。通过使用 @Configuration 和 @Bean 注解,开发者能够以编程方式创建和管理应用程序上下文中的 Bean。本文将详细介绍 @Configuration 注解的作用、如何使用它以及…

密码学中的盐值是什么?

目录 1. 盐值的基本概念 2. 盐值的作用 (1) 防止彩虹表攻击 (2) 防止相同的密码生成相同的哈希值 (3) 增加暴力破解的难度 3. 如何使用盐值&#xff1f; (1) 生成盐值 (2) 将盐值附加到密码 (3) 存储盐值和哈希值 (4) 验证密码 4. 盐值如何增加暴力破解的难度 在线暴…

基于瑞芯微RK3576国产ARM八核2.2GHz A72 工业评估板——Docker容器部署方法说明

前 言 本文适用开发环境: Windows开发环境:Windows 7 64bit、Windows 10 64bit Linux开发环境:VMware16.2.5、Ubuntu22.04.5 64bit U-Boot:U-Boot-2017.09 Kernel:Linux-6.1.115 LinuxSDK:LinuxSDK-[版本号](基于rk3576_linux6.1_release_v1.1.0) Docker是一个开…

大数据技术全解析

目录 前言1. Kafka&#xff1a;流数据的传输平台1.1 Kafka概述1.2 Kafka的应用场景1.3 Kafka的特点 2. HBase&#xff1a;分布式列式数据库2.1 HBase概述2.2 HBase的应用场景2.3 HBase的特点 3. Hadoop&#xff1a;大数据处理的基石3.1 Hadoop概述3.2 Hadoop的应用场景3.3 Hado…

mcpo的简单使用

1.安装依赖 conda create -n mcpo python3.11 conda activate mcpo pip install mcpo pip install uv2.随便从https://github.com/modelcontextprotocol/servers?tabreadme-ov-file 找一个mcp服务使用就行&#xff0c;我这里选的是爬虫 然后安装 pip install mcp-server-f…

uniapp-商城-32-shop 我的订单-订单详情和组件goods-list

上面完成了我的订单&#xff0c;通过点击我的订单中每一条数据&#xff0c;可以跳转到订单详情中。 这里就需要展示订单的状态&#xff0c;支付状态&#xff0c;物流状态&#xff0c;取货状态&#xff0c;用户信息&#xff0c;订单中的货物详情等。 1、创建一个订单详情文件 …

XCVU13P-2FHGA2104I Xilinx Virtex UltraScale+ FPGA

XCVU13P-2FHGA2104I 是 Xilinx&#xff08;现为 AMD&#xff09;Virtex UltraScale™ FPGA 系列中的高端 Premium 器件&#xff0c;基于 16nm FinFET 工艺并采用 3D IC 堆叠硅互连&#xff08;SSI&#xff09;技术&#xff0c;提供业内顶级的计算密度和带宽​。该芯片集成约 3,…

【Python3】Django 学习之路

第一章&#xff1a;Django 简介 1.1 什么是 Django&#xff1f; Django 是一个高级的 Python Web 框架&#xff0c;旨在让 Web 开发变得更加快速和简便。它鼓励遵循“不要重复自己”&#xff08;DRY&#xff0c;Don’t Repeat Yourself&#xff09;的原则&#xff0c;并提供了…

Python 设计模式:模板模式

1. 什么是模板模式&#xff1f; 模板模式是一种行为设计模式&#xff0c;它定义了一个操作的算法的骨架&#xff0c;而将一些步骤延迟到子类中。模板模式允许子类在不改变算法结构的情况下&#xff0c;重新定义算法的某些特定步骤。 模板模式的核心思想是将算法的固定部分提取…

【后端】构建简洁的音频转写系统:基于火山引擎ASR实现

在当今数字化时代&#xff0c;语音识别技术已经成为许多应用不可或缺的一部分。无论是会议记录、语音助手还是内容字幕&#xff0c;将语音转化为文本的能力对提升用户体验和工作效率至关重要。本文将介绍如何构建一个简洁的音频转写系统&#xff0c;专注于文件上传、云存储以及…

音频base64

音频 Base64 是一种将二进制音频数据&#xff08;如 MP3、WAV 等格式&#xff09;编码为 ASCII 字符串的方法。通过 Base64 编码&#xff0c;音频文件可以转换为纯文本形式&#xff0c;便于在文本协议&#xff08;如 JSON、XML、HTML 或电子邮件&#xff09;中传输或存储&#…

240422 leetcode exercises

240422 leetcode exercises jarringslee 文章目录 240422 leetcode exercises[237. 删除链表中的节点](https://leetcode.cn/problems/delete-node-in-a-linked-list/)&#x1f501;节点覆盖法 [392. 判断子序列](https://leetcode.cn/problems/is-subsequence/)&#x1f501;…

MYSQL之库的操作

创建数据库 语法很简单, 主要是看看选项(与编码相关的): CREATE DATABASE [IF NOT EXISTS] db_name [create_specification [, create_specification] ...] create_specification: [DEFAULT] CHARACTER SET charset_name [DEFAULT] COLLATE collation_name 1. 语句中大写的是…

Git Flow分支模型

经典分支模型(Git Flow) 由 Vincent Driessen 提出的 Git Flow 模型,是管理 main(或 master)和 dev 分支的经典方案: main 用于生产发布,保持稳定; dev 用于日常开发,合并功能分支(feature/*); 功能开发在 feature 分支进行,完成后合并回 dev; 预发布分支(rele…

【Spring】依赖注入的方式:构造方法、setter注入、字段注入

在Spring框架中&#xff0c;除了构造器注入&#xff08;Constructor Injection&#xff09;和Setter注入&#xff08;Setter Injection&#xff09;&#xff0c;还有一种依赖注入方式&#xff1a;字段注入&#xff08;Field Injection&#xff09;。字段注入通过在Bean的字段上…

【数学建模】随机森林算法详解:原理、优缺点及应用

随机森林算法详解&#xff1a;原理、优缺点及应用 文章目录 随机森林算法详解&#xff1a;原理、优缺点及应用引言随机森林的基本原理随机森林算法步骤随机森林的优点随机森林的缺点随机森林的应用场景Python实现示例超参数调优结论参考文献 引言 随机森林是机器学习领域中一种…

HttpSessionListener 的用法笔记250417

HttpSessionListener 的用法笔记250417 以下是关于 HttpSessionListener 的用法详解&#xff0c;涵盖核心方法、实现步骤、典型应用场景及注意事项&#xff0c;帮助您全面掌握会话&#xff08;Session&#xff09;生命周期的监听与管理&#xff1a; 1. 核心功能 HttpSessionLi…

【Python爬虫基础篇】--2.模块解析

目录 1.urllib库 1.1.request模块 1.1.1、urllib.request.urlopen() 函数 1.1.2.urllib.request.urlretrieve() 函数 1.2. error模块 1.3. parse 模块 2. BeautifulSoup4库 2.1.对象种类 2.2.对象属性 2.2.1.子节点 2.2.2.父节点 2.2.3.兄弟节点 2.2.4.回退和前进 …