0

0

什么是并发中的数据竞争(Data Race)_多线程读写非同步数据的底层后果

P粉602998670

P粉602998670

发布时间:2026-02-20 12:05:02

|

575人浏览过

|

来源于php中文网

原创

数据竞争是未定义行为的定时炸弹,非“偶尔出错”:++counter等操作因非原子性(读-改-写)和编译器重排导致结果不可预测,须用-race/-fsanitize=thread等工具检测并以锁或正确atomic语义修复。

什么是并发中的数据竞争(data race)_多线程读写非同步数据的底层后果

数据竞争不是“偶尔出错”,而是未定义行为的定时炸弹

数据竞争发生时,程序没有“大概率正确”或“多数情况正常”——只要两个线程同时读写同一个 counter 变量且没加锁,结果就不可预测。C++ 标准称其为 undefined behavior(UB),Go 的 go run -race 会直接报错,Java 的 JIT 甚至可能把整个循环优化掉。这不是 bug,是语言规范明确禁止的执行状态。

  • 现象:两个线程各执行 100 万次 ++counter,最终值可能是 102 万、189 万,也可能是 100 万——每次运行都可能不同,且不崩溃也不报错
  • 根本原因:CPU 指令不是原子的——++counter 实际是“读内存→+1→写回”三步,中间可被抢占;编译器也可能重排指令顺序
  • 常见误判:用 volatile(Java/C++)或 atomic(但用错 memory order)以为能解决,其实只是缓解可见性问题,不能保证操作原子性

怎么快速识别你代码里藏着数据竞争

别靠猜,用工具直接暴露。真实项目里,90% 的数据竞争藏在“看起来无害”的共享状态中:全局计数器、缓存 map、配置标志位、日志缓冲区。

  • Go:必须加 go run -race main.gogo test -race,它会在运行时插桩检测内存访问冲突,报错格式类似 Read at 0x00c000010240 by goroutine 7
  • C++:用 clang++ -fsanitize=thread 编译,运行时报错带线程 ID 和栈帧,比 valgrind 更准
  • Java:JVM 参数 -XX:+UnlockDiagnosticVMOptions -XX:NativeMemoryTracking=detail 辅助排查,但更推荐用 jcstress 做压力测试验证
  • 警惕伪安全写法:if (flag == true) { doWork(); flag = false; } —— flag 读和写之间没有同步,就是典型的数据竞争

修复不是加个锁就完事:锁粒度、原子类型、无锁结构怎么选

sync.Mutex 最快,但可能锁住不该锁的路径;用 std::atomic 快,但 memory_order_relaxed 在跨线程依赖场景下会失效;上无锁队列?先确认你真需要那 5% 的吞吐提升。

Musico
Musico

Musico 是一个AI驱动的软件引擎,可以生成音乐。 它可以对手势、动作、代码或其他声音做出反应。

下载
  • 优先用语言原生同步原语:sync.RWMutex(读多写少)、std::shared_mutex(C++17)、ReentrantLock(Java)
  • 简单计数/标志位:用 std::atomic_int(C++)、atomic.Int64(Go)、AtomicInteger(Java),但注意 fetch_add 才是原子自增,load+store 不行
  • 避免“锁住整个函数体”:比如在 HTTP handler 里锁住整个请求处理逻辑,实际只需保护对 userCache map 的写入
  • 无锁结构(如 boost::lockfree::queue)只适合高吞吐、低延迟且写者极少的场景;一旦出现 ABA 问题或内存回收不及时,比锁更难 debug

最容易被忽略的坑:编译器优化 + CPU 缓存一致性

你以为加了锁就万事大吉?如果锁的临界区里有未声明的共享变量、或用了 const 修饰却在别处修改,编译器可能把它当常量内联;而 CPU 缓存行伪共享(false sharing)会让两个线程频繁刷写同一 cache line,性能暴跌却不报错。

  • 检查所有被多个线程访问的变量是否显式声明为 std::atomic / volatile(仅限 Java/C# 中的内存屏障语义)/ 或受锁保护
  • 避免 struct 成员紧挨着被不同线程高频读写:struct { int a; int b; } 中若 a 被线程 1 写、b 被线程 2 读,可能因同属一个 cache line 导致争用
  • Go 中不要用 unsafe.Pointer 绕过 race detector;C++ 中 std::atomic_thread_fence 是高级工具,别为了“看起来更酷”滥用

数据竞争的修复成本,永远低于线上偶发超时、数值错乱、core dump 后花三天定位——但前提是,你在本地跑一遍 -race-fsanitize=thread,而不是等监控报警才想起这事。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1555

2023.10.24

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

823

2023.08.22

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

823

2023.08.22

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

549

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

790

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

576

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

274

2025.08.29

C++中int的含义
C++中int的含义

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

210

2025.08.29

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

776

2026.02.13

热门下载

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

精品课程

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

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