0

0

C++中如何避免数据竞争?C++多线程编程安全指南【并发陷阱】

冰火之心

冰火之心

发布时间:2025-12-18 16:42:07

|

226人浏览过

|

来源于php中文网

原创

避免数据竞争的核心是遵循互斥或无共享原则:用mutex保护共享可变状态,用atomic替代简单变量,用thread_local或不可变数据消除共享,用condition_variable/future等高级原语协作。

c++中如何避免数据竞争?c++多线程编程安全指南【并发陷阱】

避免数据竞争的核心是确保多个线程对共享数据的访问满足“互斥”或“无共享”原则——要么不让它们同时读写,要么干脆不共享可变数据。

用互斥锁保护共享可变状态

最直接的方式是用 std::mutex(或 std::shared_mutex)包裹临界区。注意:锁必须覆盖所有访问该数据的路径,包括读和写;且要避免死锁、忘记解锁、锁粒度太粗等问题。

  • 优先使用 std::lock_guardstd::unique_lock 实现 RAII 自动加锁/解锁,别手动调 lock()/unlock()
  • 多个互斥量一起加锁时,统一按地址顺序或使用 std::scoped_lock 防止死锁
  • 不要在持有锁时调用可能阻塞或调用未知第三方代码的函数(比如 I/O、回调、虚函数)

用原子操作替代简单共享变量

对单个内置类型(如 intbool、指针)的读写,若只需基本同步语义(如计数器、标志位),可用 std::atomic。它比互斥锁轻量,且无锁(lock-free)实现时性能更优。

  • 声明为 std::atomic counter{0};,用 counter.fetch_add(1) 替代 ++counter
  • 注意内存序(memory order):默认 std::memory_order_seq_cst 最安全但稍慢;高频场景可按需降级(如 relaxed 用于计数,acquire/release 用于同步临界资源)
  • 原子操作不能替代锁来保护多步逻辑(例如“先检查再修改”的复合操作),此时仍需互斥量或 std::atomic::compare_exchange_weak

消除共享:用线程局部存储或不可变数据

不共享,就无竞争。这是最彻底的方案。

短影AI
短影AI

长视频一键生成精彩短视频

下载

立即学习C++免费学习笔记(深入)”;

  • thread_local 声明每个线程独享的变量(如缓存、随机数生成器状态),注意其初始化和析构在线程生命周期内发生
  • 传递只读数据(如 const std::string&std::shared_ptr)而非可变引用,配合 std::make_shared 构造不可变对象
  • 函数式风格编程:尽量用纯函数(无副作用、不依赖外部状态)、返回新对象而非修改原对象(如用 vector new_v = old_v; + 修改,而非就地 push_back

用高级同步原语简化复杂协作

互斥锁和原子变量解决“访问控制”,但线程间协作(如等待条件、生产者-消费者)需要更高层工具

  • std::condition_variable 配合 std::unique_lock 实现条件等待,务必在 wait 的 lambda 中检查谓词(防止虚假唤醒)
  • std::promise/std::futurestd::packaged_task 适合一次性的结果传递;std::async 可启动异步任务并获取 future
  • C++17 起可用 std::shared_mutex 支持多读单写,适合读多写少场景(如配置缓存)

基本上就这些。关键不是堆砌工具,而是根据数据访问模式选对抽象:频繁读写同一块内存 → 锁;单值开关或计数 → 原子;天然隔离或只读 → 拒绝共享;需要等待或通知 → 条件变量或 future。写多线程代码时,先问自己:“这个变量,真需要被多个线程同时看到并修改吗?”

相关专题

更多
string转int
string转int

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

318

2023.08.02

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

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

524

2023.09.20

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

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

524

2023.09.20

string转int
string转int

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

318

2023.08.02

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

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

538

2024.08.29

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

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

53

2025.08.29

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

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

197

2025.08.29

string转int
string转int

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

318

2023.08.02

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

52

2026.01.19

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.9万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3.8万人学习

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

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