什么是竞争条件:Race Condition 的本质与示例

副标题 / 摘要 竞态条件是并发中最隐蔽的 bug 类型之一,源于对共享状态的非原子操作。本文给出可运行示例与规避方法。 目标读者 需要理解线程安全的工程师 做并发开发与性能优化的开发者 负责可靠性与稳定性的技术负责人 背景 / 动机 竞态条件会导致偶发错误:你很难复现,但它确实存在。 理解其成因,才能正确使用锁、原子操作与无锁结构。 核心概念 共享状态:多线程同时访问的变量 非原子操作:读-改-写不是一个不可分割的步骤 临界区:必须串行访问的代码区域 实践指南 / 步骤 识别共享状态 确定临界区 使用锁或原子操作保护 最小化临界区范围 用竞态检测工具验证 可运行示例 import threading counter = 0 def inc(): global counter for _ in range(100000): counter += 1 if __name__ == "__main__": t1 = threading.Thread(target=inc) t2 = threading.Thread(target=inc) t1.start() t2.start() t1.join() t2.join() print(counter) 在部分解释器/实现中可能输出小于 200000,这就是竞态。 解释与原理 counter += 1 并不是原子操作,它包含读取、加一、写回三个步骤。 两个线程交错执行时会丢失更新。 ...

2026年1月24日 · 1 分钟 · map[name:Jeanphilo]

为什么并发测试很难:非确定性与时序爆炸

副标题 / 摘要 并发测试困难的根源在于非确定性与时序组合爆炸。本文解释为什么难、难在哪里,并给出工程实践建议。 目标读者 做并发/多线程开发的工程师 负责可靠性与测试的团队 想理解“偶发 bug”为何难测的人 背景 / 动机 并发 bug 往往“偶发、难复现、线上才出现”。 这是因为并发调度不可预测,导致测试难以覆盖所有时序组合。 核心概念 非确定性:调度顺序不可预测 时序组合爆炸:线程交错排列数量巨大 Heisenbug:调试本身改变时序 可复现性:复现条件苛刻 实践指南 / 步骤 把并发边界缩小(减少共享状态) 引入确定性调度或模拟器 使用竞态检测工具(如 TSAN) 加大压力与重复运行(概率提高) 记录关键事件与时间线 可运行示例 下面例子展示“偶发错误”很难稳定复现: import threading x = 0 def worker(): global x for _ in range(100000): x += 1 if __name__ == "__main__": threads = [threading.Thread(target=worker) for _ in range(2)] for t in threads: t.start() for t in threads: t.join() print(x) 在某些语言/环境下会出现不一致结果,这就是并发非确定性。 解释与原理 并发执行时,读-改-写不是原子操作。 调度顺序变化会导致结果不同,而这类问题很难通过少量测试覆盖。 ...

2026年1月24日 · 1 分钟 · map[name:Jeanphilo]