深入理解Java中的volatile关键字

深入理解Java中的volatile关键字

1. 引言

在Java多线程编程中,volatile关键字扮演着重要角色。它能够保证变量的可见性和有序性,是实现线程安全的重要工具之一。本文将深入探讨volatile的实现原理,以及与之密切相关的指令重排概念。

2. volatile的作用

volatile关键字主要有两个作用:

  1. 保证变量的可见性
  2. 禁止指令重排

2.1 保证可见性

在多核处理器系统中,每个处理器都有自己的缓存。当一个线程修改了共享变量的值,这个新值可能还没有从处理器的缓存刷新到主内存,导致其他线程看不到这个新值。volatile保证,当一个线程修改了volatile变量的值,新值会立即被刷新到主内存,并且其他线程的缓存中的值会被标记为无效,强制其从主内存重新读取。

写入
刷新
失效通知
重新读取
线程1
处理器1缓存
主内存
处理器2缓存
线程2

2.2 禁止指令重排

指令重排是指处理器为了提高程序运行效率,可能会对输入代码进行优化,导致指令执行顺序发生变化。volatile关键字会在指令序列中插入内存屏障,保证特定操作不会被重排序。

3. 内存屏障

内存屏障(Memory Barrier)是一种CPU指令,用于控制特定条件下的重排序和内存可见性。Java编译器会在volatile变量的读写操作前后插入内存屏障。

3.1 内存屏障的类型

在讨论内存屏障类型之前,我们需要理解两个基本概念:

  • Load: 表示读取操作,即从主内存加载数据到CPU缓存。
  • Store: 表示写入操作,即将数据从CPU缓存写入到主内存。

有四种主要的内存屏障类型:

  1. LoadLoad屏障 (LL)
  2. StoreStore屏障 (SS)
  3. LoadStore屏障 (LS)
  4. StoreLoad屏障 (SL)

让我们详细解释每种类型:

LoadLoad屏障 (LL)
  • 含义: 确保Load1数据的装载,在Load2及所有后续装载指令的装载之前完成。
  • 作用: 防止从内存中读取的操作被重排。
StoreStore屏障 (SS)
  • 含义: 确保Store1数据对其他处理器可见(刷新到内存),在Store2及所有后续存储指令的存储之前完成。
  • 作用: 防止写入内存的操作被重排。
LoadStore屏障 (LS)
  • 含义: 确保Load1数据装载,在Store2及所有后续的存储指令刷新到内存之前完成。
  • 作用: 防止读操作与后面的写操作被重排。
StoreLoad屏障 (SL)
  • 含义: 确保Store1数据对其他处理器变得可见(刷新到内存),在Load2及所有后续装载指令的装载之前完成。
  • 作用: 是一个全能型的屏障,同时具有其他三个屏障的效果。

3.2 volatile操作中的内存屏障

在volatile变量的读写过程中,会插入不同类型的内存屏障:

线程1 缓存1 主内存 缓存2 线程2 写入volatile变量 StoreStore屏障 刷新到主内存 StoreLoad屏障 通知缓存失效 LoadLoad屏障 读取最新值 线程1 缓存1 主内存 缓存2 线程2
  1. 写操作前插入StoreStore屏障:
    • 保证在volatile写之前,所有的普通写操作已经对其他处理器可见。
  2. 写操作后插入StoreLoad屏障:
    • 保证volatile写操作对其他处理器立即可见。
  3. 读操作后插入LoadLoad屏障:
    • 保证volatile读操作后,后续所有普通读操作必须在volatile读操作完成后执行。

4. 指令重排

指令重排是指处理器为了提高程序运行效率,可能会对输入代码进行优化,导致指令执行顺序发生变化。这种重排序可能发生在以下几个层面:

  1. 编译器优化重排
  2. 指令级并行重排
  3. 内存系统重排
原始代码
编译器优化
CPU指令级优化
内存系统重排
最终执行顺序

指令重排可以提高程序的执行效率,但在多线程环境下可能导致程序出现意料之外的行为。

5. volatile如何防止指令重排

volatile通过插入内存屏障来禁止特定类型的指令重排,从而保证程序的正确性。以下是一个示例:

public class VolatileReorderingExample {private static int a = 0;private static boolean flag = false;// 使用volatile修饰flag// private static volatile boolean flag = false;public static void writer() {a = 1;                   // 1flag = true;             // 2}public static void reader() {if (flag) {              // 3if (a == 0) {        // 4System.out.println("Reordering observed!");}}}public static void main(String[] args) throws InterruptedException {for (int i = 0; i < 1000000; i++) {a = 0;flag = false;Thread writerThread = new Thread(() -> writer());Thread readerThread = new Thread(() -> reader());writerThread.start();readerThread.start();writerThread.join();readerThread.join();}System.out.println("Test completed.");}
}

注意事项:

  • 指令重排是一个微妙的现象,可能不会在每次运行或每个系统上都观察到。
  • 即使不使用 volatile,你也可能需要多次运行才能观察到重排现象。
  • 在某些现代 CPU 和 JVM 上,即使不使用 volatile,也可能很难观察到重排,因为它们可能有其他机制来保证顺序。

在这个例子中,volatile保证了2不会被重排到1之前,3不会被重排到4之后。这是通过以下方式实现的:

  1. 在操作2(写入volatile变量)之前,插入StoreStore屏障,确保1的写入操作已经完成。
  2. 在操作2之后,插入StoreLoad屏障,确保2的结果对其他线程可见。
  3. 在操作3(读取volatile变量)之后,插入LoadLoad屏障,确保后续的读操作(如4)会读取到最新的值。
StoreStore屏障
StoreLoad屏障
LoadLoad屏障
写入a
写入flag
其他操作
读取flag
读取a

这个流程图展示了如何通过内存屏障防止指令重排:

  1. StoreStore屏障确保"写入a"在"写入flag"之前完成。
  2. StoreLoad屏障确保"写入flag"的结果对其他线程立即可见。
  3. LoadLoad屏障确保在"读取a"之前,先读取到flag的最新值。

通过这种方式,volatile关键字有效地防止了可能导致程序错误的指令重排。

6. volatile的使用场景

  1. 状态标志
  2. 双重检查锁定

7. 总结

volatile是Java并发编程中的重要工具,通过保证可见性和禁止指令重排来确保程序的正确性。理解volatile的实现原理,特别是内存屏障的作用和指令重排的概念,对于编写高质量的并发程序至关重要。

参考资料

  1. 【Java面试】说一下volatile关键字

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

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

相关文章

基于人工智能及大数据的综合智能交通管理平台(可编辑30页PPT)

引言&#xff1a;随着城市化进程的加速和汽车保有量的快速增长&#xff0c;交通拥堵、交通事故频发以及交通资源分配不均等问题日益突出&#xff0c;成为制约城市发展的重要因素。为了应对这些挑战&#xff0c;基于人工智能&#xff08;AI&#xff09;及大数据技术的综合智能交…

【React】详解自定义 Hook

文章目录 一、自定义 Hook 的基本用法1. 什么是自定义 Hook&#xff1f;2. 创建自定义 Hook3. 使用自定义 Hook 二、自定义 Hook 的进阶应用1. 处理副作用2. 组合多个 Hook3. 参数化 Hook4. 条件逻辑 三、自定义 Hook 的实际应用案例1. 实现用户身份验证2. 实现媒体查询 四、最…

民大食堂用餐小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;用户管理&#xff0c;商家管理&#xff0c;档口号管理&#xff0c;商家餐品管理&#xff0c;餐品种类管理&#xff0c;购物车管理&#xff0c;订单信息管理 微信端账号功能包括&#xff1a;系统首页&a…

angular入门基础教程(七)系统路由

路由的实现 当我们系统越来复杂&#xff0c;功能越来越多&#xff0c;路由也就是必须的了。在 ng 中如何实现路由呢&#xff1f; 启用路由 在 app 目录下&#xff0c;新建一个 router 目录&#xff0c;把 app.routers.ts 文件拷贝过来&#xff0c;并修改一下。 import { Ro…

C语言程序设计16

程序设计16 问题16_1代码16_1结果16_1 问题16_2代码16_2结果16_2 问题16_3代码16_3结果16_3 问题16_1 函数 f u n fun fun 的功能是&#xff1a;逆置数组元素中的值。 例如&#xff0c;若形参 a a a 所指数组中的数据最初排列为 &#xff1a; 1 , 2 , 3 , 4 , 5 , 6 …

高职院校大数据人才培养成果导向系统构建、实施要点与评量方法

一、引言 在当今信息化快速发展的背景下&#xff0c;大数据已成为推动社会进步和产业升级的重要力量。为满足社会对大数据人才的需求&#xff0c;高职院校纷纷开设大数据相关专业&#xff0c;并致力于探索科学有效的人才培养模式。本文立足于我国信息化与智能化发展趋势&#…

【初阶数据结构】10.排序(1)

文章目录 1.排序概念及运用1.1 概念1.2 运用1.3 常见排序算法 2. 实现常见排序算法2.1 插入排序2.1.1 直接插入排序2.1.2 希尔排序2.1.2.1 希尔排序的时间复杂度计算 2.2 选择排序2.2.1 直接选择排序2.2.2 堆排序 1.排序概念及运用 1.1 概念 排序&#xff1a;所谓排序&#x…

如何用PostMan按照规律进行循环访问接口

①设置动态变量 步骤一: 设置环境变量 1. 创建环境变量集合 在 Postman 左上角选择 "环境"&#xff0c;然后点击 "添加" 来创建一个新的环境变量集合。给它起一个名称&#xff0c;比如 "uploadDemo". 2. 添加初始变量 在新创建的环境变量集…

基于python的百度迁徙迁入、迁出数据分析(三)

百度迁徙定义 百度迁徙释义&#xff1a; 百度迁徙以用户常住地所在地市或停留超过一天的非常住地定义为出发城市&#xff0c;以用户离开出发城市&#xff0c;并在非出发城市停留超过4 h以上定义为到达城市。采用4h阈值&#xff0c;排除了城际出行中的途经地。 定义参考来源…

filament 初使用记录

安装初始化 一、环境准备 官网要的 我安装的 二、下载安装 安装laravel composer create-project --prefer-dist laravel/laravel 项目名称 10.*导入 filament composer require filament/filament注册 filament 管理面板 php artisan filament:install --panels初始化…

freertos-HAL库-STM32Cubemax生成

打开cubemax选好型号配置RCC&#xff08;外部高速时钟&#xff09;这里查看原理图&#xff0c;我们把按键设为输入&#xff0c;led设为输出创建两个新任务&#xff08;default是系统创建的&#xff09;配置时钟&#xff0c;这里HSE是外部高速时钟&#xff0c;HSI是内部的&#…

axure10的安装与使用教程,问题整理

前言&#xff1a; axure10的安装与激活使用教程。 1、百度网盘下载相关资料 链接&#xff1a;https://pan.baidu.com/s/1OSD9J1wVuIptGxeRzwjlpA?pwddkbj 提取码&#xff1a;dkbj 2、开始安装&#xff0c;点击setup的安装包 除了更改地址外&#xff0c;其他的默认就行&…

Matlab编程资源库(15)数值积分

一、基本原理 求解定积分的数值方法多种多样&#xff0c;如简单的梯形法、辛普生(Simpson)法、牛顿&#xff0d;柯特斯(Newton-Cotes)法等都是经常采用的方法。它们的基本思想都是将整个积分区间[a,b]分成n个子区间[xi,xi1] &#xff0c;i1,2,…,n&#xff0c;其中 x 1a&#…

2024年PINN网络​还在火!发论文侧重点在哪儿?

2024年了&#xff0c;PINN网络依然火爆&#xff0c;各大顶会顶刊都能看见它的相关论文。 这是因为&#xff0c;AI交叉学科通常离不开求解偏微分方程PDE&#xff0c;而传统的求解方法受初始假设限制&#xff0c;一旦没设好就会导致很大的误差。 PINN作为一种新的思路&#xff…

气象水文耦合模WRF-Hydro建模技术

原文链接&#xff1a;气象水文耦合模WRF-Hydro建模技术https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247610398&idx4&sn34b4bbed4c74dcbbb0ac19ef8dcdaaff&chksmfa8271f9cdf5f8ef34ea6f721736a2fbbf8be896744ab7e46caa571c52a30628f056b4bd6964&t…

又一新AI搜索工具,OpenAI 推出新的搜索方式 SearchGPT

系列文章目录 每天推荐AI工具系列文章回顾&#xff1a; 选择 haiyi海艺图像生成、LoRA、模型的使用和训练网站 tusiart吐司艺术图像生成、LoRA 模型的使用和训练网站 解锁AI创造力的无限可能&#xff1a;探索Vivago.ai的革命性功能 文章目录 系列文章目录前言一、SearchGPT…

<数据集>手机识别数据集<目标检测>

数据集格式&#xff1a;VOCYOLO格式 图片数量&#xff1a;16172张 标注数量(xml文件个数)&#xff1a;16172 标注数量(txt文件个数)&#xff1a;16172 标注类别数&#xff1a;1 标注类别名称&#xff1a;[Phone] 使用标注工具&#xff1a;labelImg 标注规则&#xff1a;…

什么是线程安全?

什么是线程安全&#xff1f; 为什么需要线程安全&#xff1f;如何实现线程安全&#xff1f;1. 排队干活2. 自己带工具3. 用现成的安全工具 4、示例5、总结 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 在编程里&#xff0c;特别是当程序能…

推荐一款专注批量推送消息的轻量工具,支持主流平台的消息推送,简单、高效、低成本(附源码)

前言 在数字化时代&#xff0c;企业和个人面临着日益增长的消息推送需求。然而&#xff0c;现有的推送处理方案往往存在一些挑战和不足&#xff0c;如cao作复杂、成本高昂、缺乏灵活性等。这些问题不仅影响了推送效率&#xff0c;也增加了用户的负担。此外&#xff0c;随着工作…

华为od 100问 持续分享10-华为OD的面试流程细说

我是一名软件开发培训机构老师&#xff0c;我的学生已经有上百人通过了华为OD机试&#xff0c;学生们每次考完试&#xff0c;会把题目拿出来一起交流分享。 重要&#xff1a;2024年5月份开始&#xff0c;考的都是OD统一考试&#xff08;D卷&#xff09;&#xff0c;题库已经整…