volatile是什么

一、背景和问题描述

假设你写的这个多线程程序中,有两个线程:

  • 子线程(thr:把flag变量设为1,并输出“modify flag to 1”;
  • 主线程:一直在循环等待,直到flag变成1,然后退出。

代码示范:

#include <thread>
#include <iostream>int flag = 0;int main() {std::thread thr([]() {flag = 1;std::cout << "modify flag to 1" << std::endl;});while (flag == 0) {// 等待}thr.join();return 0;
}

你可能期待:

  • 子线程修改flag后,主线程马上检测到flag已变为1,然后退出。
  • 这实际上理论上没问题,但在某些环境(比如用gcc 4.8.5编译)下,结果会“卡死”,一直卡在while循环里,没人退出。

二、为什么会卡住?关键原因:编译器优化和缓存机制

这其实是一个“多线程可见性”的问题。

为什么?

  • 现代的编译器和处理器有“优化”机制:它们会试图加快程序运行速度。
  • 在没有特殊指示的情况下,编译器可能会“假定”flag在主线程中没有被别的线程改变,尤其是在没有使用同步原语的情况下。
  • 结果
    • 编译器会把flag的值“缓存”到寄存器里,读操作只在内存之前的值;
    • 导致每次循环都用“旧”的值判断(比如一直是0),不会到主存去读取最新的flag的值。

总结

  • **没有volatile时,**编译器可能会“优化”掉每次都去内存重新读取flag的操作,而只用缓存的值来判断,从而导致死循环。

三、volatile的作用

在C++中,volatile告诉编译器:“请不要对这个变量做优化,不要缓存,必须每次都从内存读取”。

改写代码:

#include <thread>
#include <iostream>volatile int flag = 0;int main() {std::thread thr([]() {flag = 1;std::cout << "modify flag to 1" << std::endl;});while (flag == 0) {// 等待}thr.join();return 0;
}

效果:

  • 通过volatile,每次while循环检测flag的值时,都会从内存中重新加载,而不是用寄存器里的“缓存值”。
  • 这样,在子线程修改了flag,主线程就能及时看到到flag==1,退出循环。

四、底层汇编分析:为什么volatile有效

这部分内容很核心,理解它可以帮你明白volatile的作用。

没有volatile时:

  • 编译器会“优化”代码,比如:
    • 只在循环开始时读取flag一次;
    • 在循环中,只用寄存器里的缓存值判断,完全避免每次都去内存读取。

用汇编表示:

  • 这样,主线程每次判断flag时,都是用一开始的值(例如0),即使子线程后来改了flag,主线程的flag值“没有变化”。

volatile时:

  • 编译器会插入“指令”,确保每次判断前,都会从内存重新读取flag的值。
  • 在汇编里表现为:每次碰到flag,都用movl(加载指令)重新加载变量的最新内容。

这样,子线程一修改flag,主线程就能立刻看到变化。


五、额外提醒:volatile的局限性

💡 volatile不是多线程同步的“护身符”

  • 它只保证“每次读写都从内存加载/存储”,但不能保证“多线程之间的同步”,或“操作的原子性”。
  • 现代多线程编程建议用**std::atomic**,它能保证:
    • 原子操作(操作步骤不可被打断);
    • 可见性(一线程修改,另一线程马上看到);
    • 内存序列一致性

总结:

  • volatile在多线程中的作用主要是阻止编译器优化变量,让变量每次都从内存重新读取。
  • 在实际多线程开发中,volatile不足以保证同步,应优先考虑std::atomic或其他同步机制。

六、总结一览

主题内容描述
volatile作用告诉编译器不要优化变量,强制每次操作都从内存中读写。
遇到的问题编译器会“缓存”读操作,导致多线程中一个线程修改的值,另一个线程看不到(死循环、程序卡死等)。
使用场景主要用于硬件状态寄存器、特殊情况的标志变量,但不替代同步工具。
更好的方案使用std::atomic保证线程安全和易维护。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/80825.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL的Docker版本,部署在ubantu系统

前言 MySQL的Docker版本&#xff0c;部署在ubantu系统&#xff0c;出现问题&#xff1a; 1.执行一个SQL&#xff0c;只有错误编码&#xff0c;没有错误提示信息&#xff0c;主要影响排查SQL运行问题&#xff1b; 2.这个问题&#xff0c;并不影响实际的MySQL运行&#xff0c;如…

专栏特辑丨悬镜浅谈开源风险治理之SBOM与SCA

随着容器、微服务等新技术日新月异&#xff0c;开源软件成为业界主流形态&#xff0c;软件行业快速发展。但同时&#xff0c;软件供应链也越来越趋于复杂化和多样化&#xff0c;软件供应链安全风险不断加剧。 软件供应链安全主要包括软件开发生命周期和软件生存运营周期&#x…

18.Excel数据透视表:第1部分创建数据透视表

一 什么是数据透视表 通过万花筒可以用不同的方式査看里面画面图像&#xff0c;在excel中可以将数据透视表看作是对准数据的万花筒&#xff0c;用不同角度去观察数据&#xff0c;也可以旋转数据&#xff0c;对数据进行重新排列&#xff0c;对大量的数据可以快速的汇总和建立交叉…

商业航天运动控制系统中的高可靠性芯片解决方案:挑战、策略与应用研究

摘要&#xff1a;随着商业航天领域的迅速发展&#xff0c;运动控制系统对芯片的可靠性提出了前所未有的挑战。本文深入探讨了商业航天运动控制系统中芯片可靠性面临的挑战&#xff0c;包括宇宙辐射效应、极端环境适应性及系统级可靠性保障等。同时&#xff0c;通过案例研究展示…

音视频学习:使用NDK编译FFmpeg动态库

1. 环境 1.1 基础配置 NDK 22b (r22b)FFmpeg 4.4Ubuntu 22.04 1.2 下载ffmpeg 官网提供了 .tar.xz 包&#xff0c;可以直接下载解压&#xff1a; wget https://ffmpeg.org/releases/ffmpeg-4.4.tar.xz tar -xvf ffmpeg-4.4.tar.xz cd ffmpeg-4.41.3 安装基础工具链 sudo …

前端开发避坑指南:React 代理配置常见问题与解决方案

前端开发避坑指南:React 代理配置常见问题与解决方案 一、为什么需要配置代理?二、使用 create-react-app 默认配置代理三、使用 http-proxy-middleware 配置复杂代理四、高级代理配置五、生产环境中的代理配置一、为什么需要配置代理? React 应用在开发过程中经常需要与后端…

用影刀RPA打通内容创作“最后一公里”:CSDN草稿一键同步多平台发布

文章目录 引言 一、需求场景&#xff1a;多平台分发的效率困境1. 痛点分析2. 影刀RPA的破局价值 二、影刀RPA是啥&#xff1f;打工人逆袭神器&#xff01;三、手把手教你造"搬运工"——技术宅的土味开发日记第一步&#xff1a;当个"偷窥狂"——观察手动操作…

进程与线程:09 进程同步与信号量

课程引入&#xff1a;进程同步与信号量 接下来这节课开始&#xff0c;我们再开始讲多进程图像。讲多进程图像的下一个点&#xff0c;前面我们讲清楚了多进程图像要想实现切换&#xff0c;调度是如何做的。同时&#xff0c;多个进程放在内存中&#xff0c;就会存在多进程合作的…

【愚公系列】《Manus极简入门》036-物联网系统架构师:“万物互联师”

&#x1f31f;【技术大咖愚公搬代码&#xff1a;全栈专家的成长之路&#xff0c;你关注的宝藏博主在这里&#xff01;】&#x1f31f; &#x1f4e3;开发者圈持续输出高质量干货的"愚公精神"践行者——全网百万开发者都在追更的顶级技术博主&#xff01; &#x1f…

MySQL 8.0 OCP 英文题库解析(四)

Oracle 为庆祝 MySQL 30 周年&#xff0c;截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始&#xff0c;将英文题库免费公布出来&#xff0c;并进行解析&#xff0c;帮助大家在一个月之内轻松通过OCP认证。 本期公布试题26~30 试题26:…

什么是原码和补码

补码的本质确实是模运算&#xff08;Modular Arithmetic&#xff09;&#xff0c;这是理解补码为何能统一加减法的核心数学原理。下面用最通俗的语言和例子解释清楚&#xff1a; —### 1. 先理解什么是“模运算”- 模运算就是“周期性计数”&#xff0c;比如钟表&#xff1a; -…

笔记项目 day02

一、用户登录接口 请求参数&#xff1a; 用loginDTO来封装请求参数&#xff0c;要加上RequestBody注解 响应参数&#xff1a; 由于data里内容较多&#xff0c;考虑将其封装到一个LoginUser的实体中&#xff0c;用户登陆后&#xff0c;需要生成jwtToken并返回给前端。 登录功…

2025年土木建筑与水利工程国际会议(ICCHE 2025)

2025 International Conference on Civil and Hydraulic Engineering (ICCHE 2025) &#xff08;一&#xff09;会议信息 会议简称&#xff1a;ICCHE 2025 大会地点&#xff1a;中国银川 投稿邮箱&#xff1a;icchesub-paper.com 收录检索&#xff1a;提交Ei Compendex,CPCI,C…

运行Spark程序-在shell中运行1

&#xff08;一&#xff09;分布式计算要处理的问题 【老师提问&#xff1a;分布式计算要面临什么问题&#xff1f;】 【老师总结】 分布式计算需要做到&#xff1a; 1.分区控制。把大的数据拆成一小份一小份的&#xff08;分区&#xff0c;分片&#xff09;让多台设备同时计算…

一文理清人工智能,机器学习,深度学习的概念

目录 一、人工智能的起源与核心范畴&#xff08;1950-1980&#xff09; 1.1 智能机器的最初构想 1.2 核心范畴的初步分化 二、机器学习的兴起与技术分化&#xff08;1980-2010&#xff09; 2.1 统计学习的黄金时代 2.2 神经网络的复兴与子集定位 2.3 技术生态的形成与AI…

《Effective Python》第1章 Pythonic 思维总结——编写优雅、高效的 Python 代码

《Effective Python》第1章 Pythonic 思维总结——编写优雅、高效的 Python 代码 在编程的世界里&#xff0c;每个语言都有其独特的风格和最佳实践。对于 Python 而言&#xff0c;“Pythonic”已经成为描述遵循 Python 特定风格的代码的代名词。这种风格不仅让代码更易读、更简…

MySQL 事务(二)

文章目录 事务隔离性理论理解隔离性隔离级别 事务隔离级别的设置和查看事务隔离级别读未提交读提交&#xff08;不可重复读&#xff09; 事务隔离性理论 理解隔离性 MySQL服务可能会同时被多个客户端进程(线程)访问&#xff0c;访问的方式以事务方式进行一个事务可能由多条SQL…

代码仓提交分支规范

以下是我部门开发时用的分支规范&#xff0c;参考于Linux社区 Tips 分支命名通常遵循一些最佳实践和规则&#xff0c;以便使分支的用途和内容清晰易懂&#xff0c;就在写一个文档的主题一样。 功能分支 (Feature Branches) 用于开发新功能。 命名格式&#xff1a;feature/功能名…

Google Earth Engine(GEE) 代码详解:批量计算_年 NDVI 并导出(附 Landsat 8 数据处理全流程)

一、代码整体目标 基于 Landsat 8 卫星数据,批量计算 2013-2020 年研究区的 NDVI(归一化植被指数),实现去云处理、数据合成、可视化及批量导出为 GeoTIFF 格式,适用于植被动态监测、生态环境评估等场景。 二、代码分步解析(含核心原理与易错点) 1. 加载并显示研究区边…

Maven 处理依赖冲突

Maven处理依赖冲突 什么是依赖冲突&#xff1f;如何解决&#xff1f;Maven自动处理依赖冲突的规则路径优先原则第一声明优先原则注意 子模块覆盖父模块父模块声明dependency子模块覆盖dependency父模块声明dependencyManagement 子模块覆盖dependency父模块声明dependencyManag…