compare_exchange_weak适合做自旋锁因其失败开销更低:在x86等架构上伪失败不触发内存屏障或总线锁,缓存压力小;返回bool并更新expected值,契合“读-改-写-重试”逻辑,需循环重试且每次重置expected为false。

compare_exchange_weak为什么适合做自旋锁
因为它的失败开销比 compare_exchange_strong 更低:在某些架构(如 x86)上,compare_exchange_weak 可能因伪失败(spurious failure)重试,但不会引发内存屏障或总线锁,对缓存行压力更小;而自旋锁本就预期短时间竞争,用 weak 版本更高效。
关键点是它返回 bool 表示 CAS 是否成功,并通过引用参数更新期望值——这正好匹配“读-改-写-重试”的自旋逻辑。
- 必须用循环包裹,因为 weak 可能假失败
- 不能依赖单次调用结果,要检查返回值并更新
expected - 初始
expected必须设为false(未加锁状态),锁变量本身用std::atomic<bool></bool>
一个最小可行的 spinlock 实现
只用 std::atomic<bool></bool> 和 compare_exchange_weak 就能写出可工作的非阻塞锁:
struct spinlock {
std::atomic<bool> locked_{false};
void lock() {
bool expected = false;
while (!locked_.compare_exchange_weak(expected, true)) {
expected = false; // 重置期望值,应对伪失败
}
}
void unlock() {
locked_.store(false, std::memory_order_release);
}
};
注意两点:
立即学习“C++免费学习笔记(深入)”;
-
lock()中每次循环都重置expected = false,否则伪失败后expected可能残留为true,导致后续 CAS 永远失败 -
unlock()用std::memory_order_release即可,配合lock()中默认的std::memory_order_seq_cst(或显式用acquire)构成 acquire-release 同步
实际使用时容易卡死的三个坑
看似简单,但掉进下面任一坑都会让线程永久自旋:
- 忘记在循环内重置
expected→ 伪失败后expected保持true,CAS 总是拿true去比false,永远失败 - 在
lock()中用了std::memory_order_relaxed→ 缺少 acquire 语义,编译器/CPU 可能把临界区代码重排到锁外 - 持有锁期间调用可能阻塞的操作(如
std::cout <<、系统调用)→ 自旋锁必须极短,否则浪费 CPU 且破坏公平性
尤其注意:C++ 标准不保证 std::atomic<bool>::compare_exchange_weak 在所有平台都真正“weak”,但行为一致——你仍得按 weak 的语义写循环,不能偷懒只试一次。
和 std::mutex 相比,什么场景才该用它
不是为了“更高级”,而是为特定低延迟、无内存分配、确定性调度的场景:
- 实时线程中禁止进入内核态(比如硬实时音频处理)
- 锁保护的临界区平均耗时
- 嵌入式环境没有完整 libc 或线程栈受限,无法承受 mutex 的内部 malloc 或 futex 等开销
一旦临界区变长、或竞争变激烈,自旋会吃光 CPU,此时 std::mutex 的休眠机制反而更优。别为了“无锁”而无锁——spinlock 仍是锁,只是不交出 CPU。
文章来自机圈观察员网,发布者:,转载请注明出处:https://www.jqgcy.com/jiquanzatan/124077.html