【Effective Modern C++】第三章 转向现代C++:16. 让const成员函数线程安全

const 成员函数承诺 "不修改对象核心逻辑状态",但常因mutable修饰的辅助变量(如多项式根的缓存rootVals、缓存有效标记rootsAreValid),在执行时修改这些变量;

多线程同时调用该 const 函数时,会出现 "不同线程读写同一块内存" 的数据竞争,导致程序行为未定义(即使逻辑上是 "读操作")。

线程安全的解决方案

1. 基础方案:使用std::mutex

  • 适用场景:需要同步多个变量 / 多个操作(如缓存值 + 缓存有效标记),保证操作的原子性;

  • 实现要点:

    • std::mutex声明为mutable(因 const 成员函数中 mutex 被视为 const 对象,而锁 / 解锁操作会修改 mutex 状态);
    • std::lock_guard自动加锁 / 解锁,避免手动解锁遗漏;
  • 副作用:std::mutex不可拷贝 / 移动,导致包含它的类也不可拷贝 / 移动;

  • 示例:多项式根缓存的线程安全实现(加锁后保证缓存计算 / 赋值的原子性)。

2. 轻量方案:使用std::atomic

  • 适用场景:仅需同步单个变量 (如函数调用次数统计),开销比 mutex 小;

  • 实现要点:将统计变量声明为std::atomic<类型>mutable,其操作是原子的(不可分割),避免数据竞争;

  • 副作用:std::atomic同样不可拷贝 / 移动,导致包含它的类不可拷贝 / 移动;

  • 坑点:绝对不能用于多个变量的同步(如缓存值 + 缓存有效标记):

    • 先赋值缓存值、后标记有效:会导致多线程重复计算(违背缓存初衷);
    • 先标记有效、后赋值缓存值:会导致线程读取到未赋值的脏数据。
方案 适用场景 优势 限制
std::mutex 多个变量 / 操作的同步 保证操作原子性 开销略大、类不可拷贝 / 移动
std::atomic 单个变量的同步 开销小、操作原子化 无法同步多个变量 / 操作

例外情况:无需保证线程安全的场景

若能绝对保证 const 成员函数永远不会在多线程环境下调用 (如仅用于单线程独占的类),可省略 mutex/atomic,避免资源开销和类拷贝 / 移动的限制;但这种场景越来越少见,优先保证线程安全。

总结

  • 保证const成员函数的线程安全性,除非可以确信它们不会用在并发语境中。
  • 运用std::atomic类型的变量会比运用互斥量提供更好的性能,但前者仅适用对单个变量或内存区域的操作。

原著在线阅读地址

相关推荐
于小猿Sup7 小时前
VMware在Ubuntu22.04驱动Livox Mid360s
linux·c++·嵌入式硬件·自动驾驶
小小编程路10 小时前
C++ 多线程与并发
java·jvm·c++
程序leo源11 小时前
Qt窗口详解
开发语言·数据库·c++·qt·青少年编程·c#
zh_xuan12 小时前
解决VS Code 控制台中文乱码
c++·vscode·乱码
郭涤生12 小时前
飞凌 RK3588 开发板同显 / 异显模式切换
c++·rk3588
计算机安禾12 小时前
【c++面向对象编程】第38篇:设计原则(二):里氏替换、接口隔离与依赖倒置
开发语言·c++
code_whiter12 小时前
C++1进阶(继承)
开发语言·c++
智者知已应修善业13 小时前
【51单片机LED闪烁10次数码管显示0-9】2023-12-14
c++·经验分享·笔记·算法·51单片机
智者知已应修善业13 小时前
【51单片机2按键控制1个敞亮LED灯闪烁和熄灭】2023-11-3
c++·经验分享·笔记·算法·51单片机
咩咦13 小时前
C++学习笔记20:日期类比较运算符重载
c++·学习笔记·类和对象·运算符重载·比较运算符·日期类