0

0

怎样在C++中实现线程安全的观察者模式 使用weak_ptr解决监听器生命周期问题

P粉602998670

P粉602998670

发布时间:2025-08-13 16:59:01

|

331人浏览过

|

来源于php中文网

原创

使用std::weak_ptr能有效解决c++++多线程下观察者模式的生命周期管理问题。1. std::weak_ptr提供弱引用,不增加引用计数,避免循环引用和对象析构阻碍;2. 通过lock()获取临时shared_ptr,确保访问时对象仍存活;3. 观察者列表需用互斥锁保护,确保添加、删除及遍历操作线程安全;4. 通知前清理失效观察者,维持列表有效性;5. 可结合shared_from_this()实现观察者在回调中安全自管理。此机制防止悬空指针,保障多线程环境下的资源安全与调用正确性。

怎样在C++中实现线程安全的观察者模式 使用weak_ptr解决监听器生命周期问题

在 C++ 多线程环境下实现观察者模式时,一个常见的问题是监听器(观察者)的生命周期管理。如果不小心,很容易出现悬空指针或访问已销毁对象的问题。使用

std::weak_ptr
是一种优雅且线程安全的解决方案。

怎样在C++中实现线程安全的观察者模式 使用weak_ptr解决监听器生命周期问题

为什么需要 weak_ptr?

在观察者模式中,被观察者通常会持有观察者的引用。如果直接使用裸指针或者

std::shared_ptr
,可能会导致以下问题:

怎样在C++中实现线程安全的观察者模式 使用weak_ptr解决监听器生命周期问题
  • 观察者无法及时从被观察者列表中移除自己;
  • 被观察者可能在其他线程中调用已经销毁的观察者方法;
  • 使用
    shared_ptr
    容易造成循环引用,影响对象析构。

std::weak_ptr
提供了一个“弱引用”机制,它不会增加引用计数,也不会阻止对象被销毁。只有在调用
lock()
获取到有效的
shared_ptr
时,才能安全地访问对象。

立即学习C++免费学习笔记(深入)”;


如何设计线程安全的观察者结构

为了确保线程安全,我们需要考虑几个关键点:

怎样在C++中实现线程安全的观察者模式 使用weak_ptr解决监听器生命周期问题
  • 被观察者持有的观察者列表必须是线程安全的;
  • 添加/删除观察者时要加锁;
  • 在通知观察者前,要检查其是否还存活;
  • 使用
    weak_ptr
    避免访问已销毁的对象;

下面是基本结构示例:

Miniflow
Miniflow

AI工作流自动化平台

下载
class Observer {
public:
    virtual void onEvent() = 0;
    virtual ~Observer() = default;
};

class Subject {
public:
    using WeakObserverPtr = std::weak_ptr;

    void addObserver(const std::shared_ptr& observer) {
        std::lock_guard lock(mtx_);
        observers_.push_back(observer);
    }

    void removeObserver(const std::shared_ptr& observer) {
        std::lock_guard lock(mtx_);
        observers_.remove_if([&](const auto& wp) {
            auto sp = wp.lock();
            return !sp || sp.get() == observer.get();
        });
    }

    void notifyObservers() {
        std::lock_guard lock(mtx_);
        for (auto it = observers_.begin(); it != observers_.end();) {
            if (auto observer = it->lock()) {
                observer->onEvent();
                ++it;
            } else {
                it = observers_.erase(it); // 自动清理失效项
            }
        }
    }

private:
    std::list observers_;
    std::mutex mtx_;
};

注意事项和优化建议

在实际使用中,有几个细节需要注意:

  • 避免频繁加锁:可以考虑使用读写锁(
    shared_mutex
    ),允许多个线程同时读取观察者列表;
  • 清理失效节点时机:可以在每次通知时顺带清理,也可以定期清理;
  • 观察者析构时自动解绑:如果希望更自动化一些,可以在观察者析构时主动触发一次 remove 操作;
  • 跨线程调用的安全性:虽然上面的代码保证了调用时对象存在,但
    onEvent()
    方法内部仍需注意线程安全;

举个例子:

struct MyObserver : public Observer, public std::enable_shared_from_this {
    void onEvent() override {
        // 可以安全访问 this
    }
};

这样就能把

shared_from_this()
weak_ptr
配合使用。


总结一下

weak_ptr
实现线程安全的观察者模式,核心在于两点:

  • weak_ptr
    存储观察者,防止访问已销毁对象;
  • 通知前调用
    lock()
    获取临时
    shared_ptr
    ,确保对象存活;
  • 所有对观察者列表的操作都要加锁;

这套机制虽然简单,但在多线程环境下非常实用,能有效规避很多潜在的资源管理问题。

基本上就这些。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

5

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

11

2026.01.21

空指针异常处理
空指针异常处理

本专题整合了空指针异常解决方法,阅读专题下面的文章了解更多详细内容。

22

2025.11.16

PHP 命令行脚本与自动化任务开发
PHP 命令行脚本与自动化任务开发

本专题系统讲解 PHP 在命令行环境(CLI)下的开发与应用,内容涵盖 PHP CLI 基础、参数解析、文件与目录操作、日志输出、异常处理,以及与 Linux 定时任务(Cron)的结合使用。通过实战示例,帮助开发者掌握使用 PHP 构建 自动化脚本、批处理工具与后台任务程序 的能力。

37

2025.12.13

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

42

2026.01.23

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

46

2026.01.23

yy漫画官方登录入口地址合集
yy漫画官方登录入口地址合集

本专题整合了yy漫画入口相关合集,阅读专题下面的文章了解更多详细内容。

202

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.5万人学习

C 教程
C 教程

共75课时 | 4.2万人学习

C++教程
C++教程

共115课时 | 13.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号