C++中锁与原子操作的区别及取舍策略
文章目录
- 锁与原子操作的基本概念
- 锁(Lock)
- 原子操作(Atomic Operations)
- 锁与原子操作的区别
- 1. **功能**
- 2. **性能**
- 3. **复杂性**
- 4. **适用场景**
- 锁与原子操作的取舍策略
- 1. **简单变量操作**
- 2. **复杂共享资源**
- 3. **性能敏感场景**
- 4. **避免死锁**
- 5. **内存顺序**
- 示例对比
- 使用锁
- 使用原子操作
- 总结
在多线程编程中,同步机制是确保线程安全的关键。C++提供了多种同步工具,其中锁(如
std::mutex
)和原子操作(如
std::atomic
)是最常用的两种。它们在功能和性能上各有特点,适用于不同的场景。本文将详细探讨锁和原子操作的区别,并提供一些关于如何选择它们的建议。
锁与原子操作的基本概念
锁(Lock)
锁是一种同步机制,用于保护共享资源,防止多个线程同时访问。C++标准库提供了多种锁的实现,如std::mutex
、std::recursive_mutex
等。使用锁时,线程在访问共享资源之前必须先获取锁,访问完成后释放锁。
std::mutex mtx;void shared_resource_access() {mtx.lock();// 访问共享资源mtx.unlock();
}
原子操作(Atomic Operations)
原子操作是一种特殊的操作,它保证操作的不可分割性,即在多线程环境下,操作不会被其他线程中断。C++11引入了std::atomic
,用于实现原子操作。
std::atomic<int> counter(0);void increment_counter() {counter.fetch_add(1, std::memory_order_relaxed);
}
锁与原子操作的区别
1. 功能
- 锁:用于保护共享资源,防止多个线程同时访问。锁的作用范围通常是一个代码块或函数。
- 原子操作:用于保证单个操作的原子性,通常用于简单的变量操作(如读取、更新、比较等)。
2. 性能
- 锁:锁的开销较大,尤其是当多个线程竞争锁时。锁的获取和释放需要系统调用,可能会导致线程阻塞和上下文切换。
- 原子操作:原子操作的开销较小,通常由硬件直接支持,性能更高。
3. 复杂性
- 锁:使用锁时需要小心避免死锁、锁顺序问题等。锁的使用较为复杂,需要合理设计锁的粒度。
- 原子操作:原子操作相对简单,不需要担心死锁问题,但需要合理选择内存顺序(如
std::memory_order
)。
4. 适用场景
- 锁:适用于保护复杂的共享资源或需要多个操作同步的场景。
- 原子操作:适用于简单的变量操作,如计数器、标志位等。
锁与原子操作的取舍策略
在选择锁和原子操作时,需要根据具体需求和场景进行权衡。以下是一些选择的建议:
1. 简单变量操作
如果需要对单个变量进行简单的操作(如读取、更新、比较等),优先选择原子操作。原子操作的性能更高,且使用起来相对简单。
std::atomic<int> counter(0);void increment_counter() {counter.fetch_add(1, std::memory_order_relaxed);
}
2. 复杂共享资源
如果需要保护复杂的共享资源(如数据结构、文件句柄等),或者需要多个操作同步完成,优先选择锁。锁可以保护整个代码块,确保线程安全。
std::mutex mtx;
std::vector<int> shared_vector;void modify_shared_vector() {mtx.lock();shared_vector.push_back(42);mtx.unlock();
}
3. 性能敏感场景
在性能敏感的场景中,尽量使用原子操作。原子操作的开销较小,不会导致线程阻塞和上下文切换。如果必须使用锁,尽量选择细粒度的锁,减少锁的持有时间。
4. 避免死锁
如果使用锁,需要特别注意避免死锁。合理设计锁的顺序,避免嵌套锁的使用。如果可能,尽量使用原子操作来简化同步机制。
5. 内存顺序
使用原子操作时,需要合理选择内存顺序。std::memory_order
提供了多种内存顺序选项,如std::memory_order_relaxed
、std::memory_order_acquire
、std::memory_order_release
等。选择合适的内存顺序可以提高性能,同时保证线程安全。
示例对比
使用锁
std::mutex mtx;
int counter = 0;void increment_counter() {mtx.lock();counter++;mtx.unlock();
}
使用原子操作
std::atomic<int> counter(0);void increment_counter() {counter.fetch_add(1, std::memory_order_relaxed);
}
在上述例子中,使用原子操作的版本性能更高,代码也更简洁。但如果需要保护一个复杂的数据结构,锁可能是更好的选择。
总结
锁和原子操作是C++中两种重要的同步机制,各有优缺点。锁适用于保护复杂的共享资源,原子操作适用于简单的变量操作。在选择时,需要根据具体需求、性能要求和代码复杂性进行权衡。以下是一些选择的要点:
- 简单变量操作:优先选择原子操作。
- 复杂共享资源:优先选择锁。
- 性能敏感场景:优先选择原子操作。
- 避免死锁:合理设计锁的使用,尽量使用原子操作简化同步。
希望本文的介绍和建议能够帮助你在多线程编程中更好地选择锁和原子操作。如果你对某个具体场景有疑问,欢迎在评论区留言讨论。
相关文章:

C++中锁与原子操作的区别及取舍策略
文章目录 锁与原子操作的基本概念锁(Lock)原子操作(Atomic Operations) 锁与原子操作的区别1. **功能**2. **性能**3. **复杂性**4. **适用场景** 锁与原子操作的取舍策略1. **简单变量操作**2. **复杂共享资源**3. **性能敏感场景…...
ESP32对接巴法云实现配网
目录 序言准备工作巴法云注册与使用Arduino准备 开发开始配网 序言 本文部分内容摘抄原创作者巴法云-做优秀的物联网平台 代码有部分修改并测试运行正常 巴法云支持免费用户通过开发对接实现各智能音箱设备语音控制智能家居设备,并有自己的App进行配网和控制&…...
《深度剖析:基于Meta的GameFormer构建自博弈AI游戏代理》
自博弈AI游戏代理,是一种具备自主学习和自我提升能力的人工智能系统。它打破了传统AI依赖预设规则和固定策略的局限,能够在游戏过程中不断与自身进行对战,通过反复博弈来积累经验、优化策略,从而实现智能水平的持续提升 。这种独特…...

C++语法系列之类型转换
前言 类型转换是经常存在的情况,类型转换分为隐式类型转化 和 显式类型转化 隐式类型转化:编译器在编译阶段自动进行,能转就转,不能转就编译失败 double i 3.3; int b i; //隐式类型转化 double -> intC搞出来了四种强制类…...
Qwen3 技术报告解读一
📘 Qwen3 技术报告解读:通义千问系列新成员的技术亮点与能力分析 一、论文写了什么? 本文来自阿里通义实验室发布的 《Qwen3 Technical Report》,介绍了其最新一代大语言模型 Qwen3 的技术架构、训练方法以及在多个关键任务上的…...

详解开漏输出和推挽输出
开漏输出和推挽输出 以上是 GPIO 配置为输出时的内部示意图,我们要关注的其实就是这两个 MOS 管的开关状态,可以组合出四种状态: 两个 MOS 管都关闭时,输出处于一个浮空状态,此时他对其他点的电阻是无穷大的ÿ…...

【八股消消乐】索引失效与优化方法总结
😊你好,我是小航,一个正在变秃、变强的文艺倾年。 🔔本专栏《八股消消乐》旨在记录个人所背的八股文,包括Java/Go开发、Vue开发、系统架构、大模型开发、具身智能、机器学习、深度学习、力扣算法等相关知识点ÿ…...

一步一步配置 Ubuntu Server 的 NodeJS 服务器详细实录——4. 配置服务器终端环境 zsh , oh my zsh, vim
前言 通过前面几篇文章,我们顺利的 安装了 ubuntu server 服务器,并且配置好了 ssh 免密登录服务器,也安装好了 服务器常用软件安装,接下来,我们要仔细的配置一下我们的终端环境,让服务器的终端更加好用。 一般情况下…...

数据安全合规体系构建的“三道防线“
引言 "三道防线"模型架构图 #mermaid-svg-wbeppAbwa3Vb3nL2 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-wbeppAbwa3Vb3nL2 .error-icon{fill:#552222;}#mermaid-svg-wbeppAbwa3Vb3nL2 .error-text{fi…...

【Spring底层分析】Spring AOP基本使用+万字底层源码阅读分析
一、AOP基本使用 三步: 将业务逻辑组件和切面类都加入到容器中,告诉Spring哪个是切面类(Aspect)在切面类上的每一个通知方法上标注通知注解,告诉Spring何时(Before、After、Around……)何地运…...
Python数据分析及可视化中常用的6个库及函数(二)
Python数据分析及可视化中常用的6个库及函数(二) 摘要:以下是Python数据分析及可视化常用的6个库的详细介绍,包括它们的概述以及每个库中最常用的10个函数(如果某些库常用函数不足10个,则列出所有常用函数)。每个函数都附带功能描述、用法说明和使用示例。这些库…...

新德通科技:以创新驱动光通信一体化发展,赋能全球智能互联
在数字经济与AI技术高速发展的今天,光通信作为信息传输的核心基础设施,正迎来前所未有的升级浪潮。深圳新德通科技有限公司(以下简称“新德通科技”)凭借其深厚的技术积累与一体化产品布局,成为行业内的中坚力量。本文…...
Selenium的底层原理
Selenium 底层主要依赖于 WebDriver 协议(即 W3C WebDriver 规范,早期也有 JSON Wire Protocol)来实现对浏览器的远程控制,其核心架构可以分为以下几层: Selenium 客户端(Client Library) 支持多…...
PostgreSQL的扩展 auth_delay
PostgreSQL的扩展 auth_delay auth_delay 是 PostgreSQL 提供的一个安全相关扩展,主要用于防止暴力破解攻击。它通过在认证失败后引入人为延迟来增加暴力破解的难度。 一、扩展基础 功能:在认证失败后增加延迟目的:减缓暴力破解和字典攻击…...
[Java 基础]Java 是什么
Java 是一门编程语言。 查看编程语言热门排行:https://www.tiobe.com/tiobe-index/ Java 的特点: 面向对象:Java 是面向对象的语言,支持封装、继承和多态等特性。 平台无关性:Java 通过“一次编写,到处…...
Qt学习2
跟学视频 1.菜单栏和工具栏 //菜单栏最多只能有一个//菜单栏创建QMenuBar * bar menuBar();//将菜单栏放到窗口中setMenuBar(bar);//创建菜单QMenu * fileMenu bar->addMenu("开始");QMenu * editMenu bar->addMenu("编辑");//创建菜单项QAction…...

C++ 内存泄漏检测器设计
文章目录 1. C中的动态内存分配2. 什么是内存泄漏3. 内存泄漏的代码案例4. 内存泄漏检查器的设计模块1:位置信息捕获:模块2:内存分配跟踪:模块3:内存释放跟踪:模块4:泄漏记录存储:模…...
在 Linux 上安装 Nmap 工具
📦 在 Linux 上安装 Nmap 工具指南 Nmap(Network Mapper)是功能强大的网络扫描工具,以下是各种 Linux 发行版的安装方法: 🧩 通用安装方法 1. 使用包管理器安装(推荐) # Debian/…...
从零打造AI面试系统全栈开发
🤖 AI面试系统开发完整教程 📋 项目概述 本教程将带你从零开始构建一个完整的AI面试系统,包含前端、后端、AI集成和部署的全流程。 源码地址 技术栈 前端: React TypeScript Vite Vaadin Components后端: Spring Boot Spring Securi…...

破局与进阶:ueBIM 在国产 BIM 赛道的差距认知与创新实践
作为国产BIM领域的探索者,斯维尔ueBIM自诞生以来始终以追赶国际头部技术为目标,但不可否认的是,在核心功能覆盖、行业生态成熟度以及全球市场占有率等方面,我们与Autodesk Revit、Bentley Systems等国际巨头仍存在显著差距。这种差…...

分布式流处理与消息传递——向量时钟 (Vector Clocks) 算法详解
Java 实现向量时钟 (Vector Clocks) 算法详解 一、向量时钟核心原理 #mermaid-svg-JcZ1GT0r1ZNSy6W7 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-JcZ1GT0r1ZNSy6W7 .error-icon{fill:#552222;}#mermaid-svg-JcZ…...

20250603在荣品的PRO-RK3566开发板的Android13下的命令行查看RK3566的温度
20250603在荣品的PRO-RK3566开发板的Android13下的命令行查看RK3566的温度 2025/6/3 11:58 RK3566的cpu运行效率 top rk3566_t:/ # rk3566_t:/ # rk3566_t:/ # cd /sys/class/thermal/ rk3566_t:/sys/class/thermal # ls -l rk3566_t:/sys/class/thermal # cd thermal_zone0/ r…...

帝可得 - 设备管理
一. 需求说明 设备管理主要涉及到三个功能模块,业务流程如下: 新增设备类型: 允许管理员定义新的售货机型号,包括其规格和容量。 新增设备: 在新的设备类型定义后,系统应允许添加新的售货机实例,并将它们分配到特定的…...
FTXUI配置
对于 FTXUI 的安装与配置, 官方已经给出了三种方案. 第一种: 使用 FetchContent 远程拉取第二种: 在你本地安装 FTXUI 库, 然后通过 find_package 使用第三种: 使用 Git 子模块 FetchContent 无需手动下载安装 FTXUI, 通过 CMake 自动从 GitHub 拉取并编译依赖 include(Fet…...
Caliper压力测试
目前FISCO BCOS适配的Caliper版本为0.2.0,请在部署Caliper运行环境时确保Caliper的版本为0.2.0,如在部署或使用过程中遇到任何问题,请优先参考 https://github.com/FISCO-BCOS/FISCO-BCOS/issues/1248 中的解决方案进行排查。 1. 环境要求 …...

【iOS安全】使用LLDB调试iOS App | LLDB基本架构 | LLDB安装和配置
LLDB基本架构 参考: https://crifan.github.io/ios_re_dynamic_debug/website/debug_code/lldb_debugserver.html https://book.crifan.org/books/ios_re_debug_debugserver_lldb/website/ LLDB安装和配置 1. 让iPhone中出现/Developer/usr/bin/debugserver 最初…...
一、核心概念深入解析
一、核心概念深入解析 1. shared_ptr 的线程安全性澄清 引用计数是原子操作:shared_ptr 的引用计数(use_count)在多线程中递增 / 递减是安全的(原子操作),但对象本身的读写需额外同步(如 std:…...
python直方图
在Python中,绘制直方图(Histogram)是一项非常常见的任务,通常用于数据可视化,以展示数据的分布情况。Python中有多种库可以绘制直方图,其中最常用的两个库是Matplotlib和Seaborn。此外,Pandas库…...
[特殊字符] Unity 性能优化终极指南 — Text / TextMeshPro 组件篇
UGUI Text组件的不当使用及其性能瓶颈与优化 在Unity UGUI系统中,Text 组件(或其升级版 TextMeshPro)是显示文本信息的核心元素。然而,如果不当使用,它极易成为UI性能瓶颈的罪魁祸首,尤其是在预制体、属性…...

Idea 配置 Maven 环境
下载 Maven 官网:https://maven.apache.org/index.html 点击左侧 Downloads,然后选择 Files 中的 zip 包下载(下载慢可以使用迅雷) 配置 Maven 将压缩包解压,比如我解压后放到了 D:\developer\environment\apache-…...