您的位置:首页 >C++高性能观察者模式实现方法
发布于2026-03-17 阅读(0)
扫一扫,手机访问

std::function + std::vector 存回调,别手写链表性能瓶颈常出在通知阶段的遍历开销和内存碎片上。手写双向链表看似“可控”,实际带来指针跳转、缓存不友好、删除时迭代器失效等问题。现代 C++ 更推荐用连续内存容器存 std::function 对象。
实操建议:
std::vector 存观察者,插入/通知都是 O(1) 平摊复杂度std::array 预分配,省去动态扩容判断std::function 的构造开销:捕获大对象或绑定复杂逻辑时,考虑用 std::shared_ptr 包一层再传入,避免拷贝std::shared_ptr 管理观察者生命周期,别裸指针 + weak_ptr 检查裸指针观察者容易悬空,而每轮通知前都调 lock() 再判空,会把热点路径拖慢 20%+(实测 clang 15 / x86-64)。更轻量的做法是让观察者自己管理生命周期,发布者只持 std::shared_ptr。
实操建议:
std::enable_shared_from_this,注册时传 shared_from_this()std::shared_ptr,通知时直接调用成员函数on_destroy 回调)weak_ptr,但务必把 lock() 移到注册/注销路径,而非通知路径90% 的误以为“观察者模式必须线程安全”,结果给每个 notify() 加 std::mutex,吞掉 3~5 倍吞吐。真实情况是:线程模型决定同步策略,不是模式本身决定。
实操建议:
std::vector 读写都在同一线程,安全且最快std::queue + std::mutex 收集事件,通知仍在单线程执行thread_local 或 std::unordered_map 分离,避免争用std::function 回调里做耗时操作——这会让整个通知卡住,应转成异步投递std::move 传事件对象,但别 move 所有东西事件对象如果含大字段(如 std::vector 日志内容),按值传参会触发深拷贝;但盲目所有参数都 std::move,可能破坏 const 正确性或导致多次 move 后访问未定义行为。
实操建议:
[[nodiscard]],并禁用拷贝构造,强制移动语义void notify(Event&& e),内部用 std::move(e) 转发给各回调e.type),提供 const 引用重载:void notify(const Event& e),避免无谓移动e 在 notify 函数内不可再访问——常见坑是 move 后还打印日志或校验字段最易被忽略的是事件对象的内存布局和对齐。如果事件里混用 bool、double、std::string,缓存行浪费比你预想的严重。高频路径上,优先用 POD 结构 + 手动 padding,别依赖编译器优化。
上一篇:Edge浏览器开启密码保存教程
下一篇:4S店销售技巧:快速成交秘诀
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9