第13章 深入volatile关键字(Java高并发编程详解:多线程与系统设计)
1.并发编程的三个重要特性
并发编程有三个至关重要的特性,分别是原子性、有序性和可见性
1.1 原子性
所谓原子性是指在一次的操作或者多次操作中,要么所有的操作全部都得到了执行并
且不会受到任何因素的干扰而中断,要么所有的操作都不执行。
注意:两个原子性的操作结合在一起未必还是原子性的,比如i++(其中get i,i+1和set i=x三者皆是原子性操作,但是不代表i++就是原子性操作)。volatile关键字不保证数据的原子性,synchronized关键字保证,自JDK 1.5版本起,其提供的原子类型变量也可以保证原子性。
1.2 可见性
可见性是指,当一个线程对共享变量进行了修改,那么另外的线程可以立即看到修改后的最新值
1.3 有序性
所谓有序性是指程序代码在执行过程中的先后顺序, 由于Java在编译器以及运行期的优化,导致了代码的执行顺序未必就是开发者编写代码时的顺序,比如:
int x=10;
in ty=0;
x++;
y=20;
对于这段代码有可能它的执行顺序就是代码本身的顺序,有可能发生了重排序导致int y=0优先于int x=10执行,但是绝对不可能出现y=x+1优先于x++执行的执行情况,如果一个指令x在执行的过程中需要用到指令y的执行结果,那么处理器会保证指令y在指令x之前执行,这就好比y=x+1执行之前肯定要先执行x++一样。
2.JMM三如何保证大特性
Java的内存模型规定了所有的变量都是存在于主内存(RAM) 当中的, 而每个线程都有自己的工作内存或者本地内存(这一点很像CPU的Cache) , 线程对变量的所有操作都必须在自己的工作内存中进行,而不能直接对主内存进行操作,并且每一个线程都不能访问其他线程的工作内存或者本地内存。
2.1 JMM与原子性
在Java语言中,对基本数据类型的变量读取赋值操作都是原子性的,对引用类型的变量读取和赋值的操作也是原子性的,因此诸如此类的操作是不可被中断的,要么执行,要么不执行,正所谓一荣俱荣一损俱损。
不过话虽如此简单,但是理解起来未必不会出错,下面我们就来看几个例子:
(1)x=10;赋值操作
x=10的操作是原子性的,执行线程首先会将x=10写人工作内存中,然后再将其写入主内存(有可能在往主内存进行数值刷新的过程中其他线程也在对其进行刷新操作,比如另外一个线程将其写为11,但是最终的结果肯定要么是10,要么是11,不可能出现其他情况,单就赋值语句这一点而言其是原子性的)。
(2)y=x; 赋值操作
这条操作语句是非原子性的,因为它包含如下两个重要的步骤。
1)执行线程从主内存中读取x的值(如果x已经存在于执行线程的工作内存中,则直接获取)然后将其存人当前线程的工作内存之中。
2)在执行线程的工作内存中修改y的值为x,然后将y的值写入主内存之中。虽然第一步和第二步都是原子类型的操作,但是合在一起就不是原子操作了。
(3)y++;自增操作
这条操作语句是非原子性的,因为它包含三个重要的步骤,具体如下。
1)执行线程从主内存中读取y的值(如果y已经存在于执行线程的工作内存中,则直接获取),然后将其存人当前线程的工作内存之中。
2)在执行线程工作内存中为y执行加1操作。
3)将y的值写人主内存。
(4)z = z+1; 加一操作(与自增操作等价)
这条操作语句是非原子性的,因为它包含三个重要的步骤,具体如下。
1)执行线程从主内存中读取z的值(如果z已经存在于执行线程的工作内存中,则直接获取),然后将其存人当前线程的工作内存之中。
2)在执行线程工作内存中为z执行加1操作。
3)将z的值写入主内存。
由此我们可以得出以下几个结论。
- 多个原子性的操作在一起就不再是原子性操作了。
- 简单的读取与赋值操作是原子性的,将一个变量赋给另外一个变量的操作不是原子性的。
- Java内存模型(JMM) 只保证了基本读取和赋值的原子性操作, 其他的均不保证,如果想要使得某些代码片段具备原子性, 需要使用关键字synchronized, 或者JUC中的lock。如果想要使得int等类型自增操作具备原子性, 可以使用JUC包下的原子封装类型java.util.concurrent.atomic.*
总结:volatile关键字不具备保证原子性的语义
2.2 JMM与可见性
在多线程的环境下,如果某个线程首次读取共享变量,则首先到主内存中获取该变量,然后存入工作内存中,以后只需要在工作内存中读取该变量即可。同样如果对该变量执行了修改的操作,则先将新值写入工作内存中,然后再刷新至主内存中。但是什么时候最新的值会被刷新至主内存中是不太确定的, 这也就解释了为什么Volatile Foo中的Reader线程始终无法获取到in it value最新的变化。
Java提供了以下三种方式来保证可见性。
- 使用关键字volatile, 当一个变量被volatile关键字修饰时, 对于共享资源的读操作会直接在主内存中进行(当然也会缓存到工作内存中,当其他线程对该共享资源进行了修改,则会导致当前线程在工作内存中的共享资源失效,所以必须从主内存中再次获取),对于共享资源的写操作当然是先要修改工作内存,但是修改结束后会立刻将其刷新到主内存中。
- 通过synchronized关键字能够保证可见性, synchronized关键字能够保证同一时刻只有一个线程获得锁,然后执行同步方法,并且还会确保在锁释放之前,会将对变量的修改刷新到主内存当中。
- 通过JUC提供的显式锁Lock也能够保证可见性, Lock的lock方法能够保证在同一时刻只有一个线程获得锁然后执行同步方法, 并且会确保在锁释放(Lock的unlock方法)之前会将对变量的修改刷新到主内存当中。
总结:volatile关键字具有保证可见性的语义。
2.3 JMM与有序性
在Java的内存模型中, 允许编译器和处理器对指令进行重排序, 在单线程的情况下,重排序并不会引起什么问题,但是在多线程的情况下,重排序会影响到程序的正确运行,Java提供了三种保证有序性的方式, 具体如下。
- 使用volatile关键字来保证有序性。
- 使用synchronized关键字来保证有序性。
- 使用显式锁Lock来保证有序性。
此外,Java的内存模型具备一些天生的有序性规则,不需要任何同步手段就能够保证有序性,这个规则被称为Happens-before原则。如果两个操作的执行次序无法从happens-before原则推导出来,那么它们就无法保证有序性, 也就是说虚拟机或者处理器可以随意对它们进行重排序处理。
下面我们来具体看看都有哪些happens-before原则。
- 程序次序规则:在一个线程内,代码按照编写时的次序执行,编写在后面的操作发生于编写在前面的操作之后。
这句话的意思看起来是程序按照编写的顺序来执行,但是虚拟机还是可能会对程序代码的指令进行重排序,只要确保在一个线程内最终的结果和代码顺序执行的结果一致即可。
- 锁定规则:一个unlock操作要先行发生于对同一个锁的lock操作。
这句话的意思是,无论是在单线程还是在多线程的环境下,如果同一个锁是锁定状态,那么必须先对其执行释放操作之后才能继续进行lock操作。
- volatile变量规则:对一个变量的写操作要早于对这个变量之后的读操作。
根据字面的意思来理解是, 如果一个变量使用volatile关键字修饰, 一个线程对它进行读操作,一个线程对它进行写操作,那么写入操作肯定要先行发生于读操作,关于这个规则我们在3.3节中还会继续介绍。
- 传递规则:如果操作A先于操作B,而操作B又先于操作C,则可以得出操作A肯定要先于操作C, 这一点说明了happens-before原则具备传递性。
- 线程启动规则:Thread对象的start() 方法先行发生于对该线程的任何动作, 这也是我们在第一部分中讲过的, 只有start之后线程才能真正运行, 否则Thread也只是一个对象而已。
- 线程中断规则:对线程执行interrupt() 方法肯定要优先于捕获到中断信号, 这句话的意思是指如果线程收到了中断信号, 那么在此之前势必要有interrupt() 。
- 线程的终结规则:线程中所有的操作都要先行发生于线程的终止检测,通俗地讲,线程的任务执行、逻辑单元执行肯定要发生于线程死亡之前。
- 对象的终结规则:一个对象初始化的完成先行发生于finalize() 方法之前, 这个更没什么好说的了,先有生后有死。
总结: volatile关键字具有保证顺序性的语义
3. volatile关键字深入解析
3.1volatile关键字的语义
被volatile修饰的实例变量或者类变量具备如下两层语义。
- 保证了不同线程之间对共享变量操作时的可见性, 也就是说当一个线程修改volatile修饰的变量,另外一个线程会立即看到最新的值。
- 禁止对指令进行重排序操作。
(1)理解volatile保证可见性
关于共享变量在多线程间的可见性, 在Volatile Foo例子中已经体现得非常透彻了,Updater线程对in it_value变量的每一次更改都会使得Reader线程能够看到(在happens-before规则中, 第三条volatile变量规则:对一个变量的写操作要早于对这个变量之后的读操作),其步骤具体如下。
1) Reader线程从主内存中获取in it_value的值为0, 并且将其缓存到本地工作内存中。
2) Updater线程将in it_value的值在本地工作内存中修改为1, 然后立即刷新至主内
存中。
3) Reader线程在本地工作内存中的in it_value失效(反映到硬件上就是CPU的L 1或
者L 2的CacheLine失效) 。
4) 由于Reader线程工作内存中的in it_value失效, 因此需要到主内存中重新读取in it
value的值。
(2)理解volatile保证顺序性
volatile关键字对顺序性的保证就比较霸道一点, 直接禁止JVM和处理器对volatile关键字修饰的指令重排序, 但是对于volatile前后无依赖关系的指令则可以随便怎么排序,比如
int x= 0;
int y= 1;
volatile int z = 20;
x++;
Y--;
在语句volatile in tz=20之前, 先执行x的定义还是先执行y的定义, 我们并不关心,只要能够百分之百地保证在执行到z=20的时候x=0,y=1,同理关于x的自增以及y的自减操作都必须在z=20以后才能发生。
(3) 理解volatile不保证原子性
i++的操作其实是由三步组成的,具体如下。
1)从主内存中获取i的值,然后缓存至线程工作内存中。
2)在线程工作内存中为i进行加1的操作。
3)将i的最新值写入主内存中。
上面三个操作单独的每一个操作都是原子性操作,但是合起来就不是,因为在执行的中途很有可能会被其他线程打断,例如如下操作情况。
1)假设此时i的值为100,线程A要对变量i执行自增操作,首先它需要到主内存中读取i的值, 可是此时由于CPU时间片调度的关系, 执行权切换到了线程B, A线程进入了RUNNABLE状态而不是RUNNING状态。
2)线程B同样需要从主内存中读取i的值,由于线程A没有对i做过任何修改操作,因此此时B获取到的i仍然是100。
3)线程B工作内存中为i执行了加1操作,但是未刷新至主内存中。
4) CPU时间片的调度又将执行权给了线程A, A线程直接对工作线程中的100进行加1运算(因为A线程已经从主内存中读取了i的值),由于B线程并未写人i的最新值,因此A线程工作空间中的100不会被失效。
5)线程A将i=101写人主内存之中。
6)线程B将i=101写入到主内存中。
3.2 volatile的原理和实现机制
通过对Open JDK下unsafe.cpp源码的阅读, 会发现被volatile修饰的变量存在于一个“1ock; ”的前缀, 源码如下:
“lock; ”前缀实际上相当于是一个内存屏障, 该内存屏障会为指令的执行提供如下几个保障。
- 确保指令重排序时不会将其后面的代码排到内存屏障之前。
- 确保指令重排序时不会将其前面的代码排到内存屏障之后。
- 确保在执行到内存屏障修饰的指令时前面的代码全部执行完成。
- 强制将线程工作内存中值的修改刷新至主内存中。
- 如果是写操作, 则会导致其他线程工作内存(CPU Cache) 中的缓存数据失效。
3.3 volatile的使用场景
(1)开关控制利用可见性的特点
(2)状态标记利用顺序性特点
(3)Singleton设计模式的double-check也是利用了顺序性特点
3.4 volatile和synchronized
通过对volatile关键字的学习和之前对synchronized关键字的学习, 我们在这里总结一下两者之间的区别。
(1)使用上的区别
- volatile关键字只能用于修饰实例变量或者类变量, 不能用于修饰方法以及方法参数和局部变量、常量等。
- synchronized关键字不能用于对变量的修饰, 只能用于修饰方法或者语句块。
- volatile修饰的变量可以为null, synchronized关键字同步语句块的monitor对象不能为null。
(2)对原子性的保证
- volatile无法保证原子性。
- 由于synchronized是一种排他的机制, 因此被synchronized关键字修饰的同步代码是无法被中途打断的,因此其能够保证代码的原子性。
(3) 对可见性的保证
- 两者均可以保证共享资源在多线程间的可见性,但是实现机制完全不同。
- synchronized借助于JVM指令monitor enter和monitor exit对通过排他的方式使得同步代码串行化, 在monitor exit时所有共享资源都将会被刷新到主内存中。
- 相比较于synchronized关键字volatile使用机器指令(偏硬件) “lock; ”的方式迫使其他线程工作内存中的数据失效,不得到主内存中进行再次加载。
(4) 对有序性的保证
- volatile关键字禁止JVM编译器以及处理器对其进行重排序, 所以它能够保证有序性。
- 虽然synchronized关键字所修饰的同步方法也可以保证顺序性, 但是这种顺序性是以程序的串行化执行换来的, 在synchronized关键字所修饰的代码块中代码指令也会发生指令重排序的情况,比如:
synchronized(this) {int x=10;int y=20;x++;y=y+1;
}
(5) 其他
- volatile不会使线程陷入阻塞。
- synchronized关键字会使线程进人阻塞状态。
相关文章:

第13章 深入volatile关键字(Java高并发编程详解:多线程与系统设计)
1.并发编程的三个重要特性 并发编程有三个至关重要的特性,分别是原子性、有序性和可见性 1.1 原子性 所谓原子性是指在一次的操作或者多次操作中,要么所有的操作全部都得到了执行并 且不会受到任何因素的干扰而中断,要么所有的操作都不执行…...

[STM32 标准库]定时器输出PWM配置流程 PWM模式解析
前言: 本文内容基本来自江协,整理起来方便日后开发使用。MCU:STM32F103C8T6。 一、配置流程 1、开启GPIO,TIM的时钟 /*开启时钟*/RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); //开启TIM2的时钟RCC_APB2PeriphClockC…...

web3py+flask+ganache的智能合约教育平台
最近在学习web3的接口文档,使用web3pyflaskganache写了一个简易的智能合约教育平台,语言用的是python,ganche直接使用的本地区块链网络,用web3py进行交互。 代码逻辑不难,可以私信或者到我的闲鱼号夏沫mds获取我的代码…...

< OS 有关 > 阿里云:轻量应用服务器 的使用 :轻量化 阿里云 vpm 主机
原因: < OS 有关 > 阿里云:轻量应用服务器 的使用 :从新开始 配置 SSH 主机名 DNS Tailscale 更新OS安装包 最主要是 清除阿里云客户端这个性能杀手-CSDN博客 防止 I/O 祸害系统 操作: 查看进程&#x…...

【技术】TensorRT 10.7 安装指南(Ubuntu22.04)
原文链接:https://mengwoods.github.io/post/tech/008-tensorrt-installation/ 本文安装的版本如下: Ubuntu 22.04 Nvidia Driver 538.78 CUDA 12.2 cuDNN 8.9.7 TensorRT 10.7 安装前的准备(可选) 在安装新版本之前…...

Linux 权限管理
hello!这里是敲代码的小董,很荣幸您阅读此文,本文只是自己在学习Linux过程中的笔记,如有不足,期待您的评论指点和关注,欢迎欢迎~~ ✨✨个人主页:敲代码的小董 💗💗系列专…...

8.2 从看图识字到智能解读:GPT-4 with Vision 开启多模态 AI 新纪元
从看图识字到智能解读:GPT-4 with Vision 开启多模态 AI 新纪元 引言:AI 的多模态跃迁 随着人工智能技术的快速发展,我们正迈入一个新的智能交互时代。传统的 AI 模型主要聚焦于文本处理,而多模态 AI 模型如 GPT-4 with Vision(GPT-4V) 则能够同时处理图像和文本。GPT-4…...

差分轮算法-两个轮子计算速度的方法-阿克曼四轮小车计算方法
四轮驱小车的话: 转向角度计算方法:float turning_angle z_angular / x_linear; // 转向角度,单位为弧度 速度的话直接用线速度 两轮驱动小车: 计算公式: leftSpeed x_linear - z_angular * ORIGINBOT_WHEEL_TRACK /…...

使用.NET 8构建高效的时间日期帮助类
使用.NET 8构建高效的时间日期帮助类 在现代Web应用程序中,处理日期和时间是一个常见的需求。无论是记录日志、生成报告还是进行数据分析,正确处理日期和时间对于确保数据的准确性和一致性至关重要。本文将详细介绍如何使用ASP.NET Core和C#构建一个高效…...

学习std::is_base_of笔记
1、std::is_base_of简介 在现代 C 中,模板元编程(Template Metaprogramming)是一种非常强大的编程技巧,它让我们能够在编译期进行类型推导和约束。而 std::is_base_of 是一个重要的工具,可以用来检查一个类型是否是另…...

第 25 场 蓝桥月赛
3.过年【算法赛】 - 蓝桥云课 问题描述 蓝桥村的村民们正准备迎接新年。他们计划宰杀 N 头猪,以庆祝一整年的辛勤劳作和丰收。每头猪的初始位置位于下标 xi,所有 xi 均为偶数,保证没有两头猪初始位置相同。 当猪意识到人类打算宰杀它们…...

【设计模式-行为型】访问者模式
一、什么是访问者模式 说起来访问者模式,其实很少用。我一直在思考该用什么样的例子把这个设计模式表述清晰,最近突然想到一个例子也许他就是访问者。港片有过很辉煌的年代,小的时候一直在看港片觉得拍的非常好,而且演员的演技也在…...

无人机微波图像传输数据链技术详解
无人机微波图像传输数据链技术是无人机通信系统中的关键组成部分,它确保了无人机与地面站之间高效、可靠的图像数据传输。以下是对该技术的详细解析: 一、技术原理 无人机微波图像传输数据链主要基于微波通信技术实现。在数据链路中,图像数…...

SpringCloud系列教程:微服务的未来(十七)监听Nacos配置变更、更新路由、实现动态路由
前言 在微服务架构中,API 网关是各个服务之间的入口点,承担着路由、负载均衡、安全认证等重要功能。为了实现动态的路由配置管理,通常需要通过中心化的配置管理系统来实现灵活的路由更新,而无需重启网关服务。Nacos 作为一个开源…...

【QT】 控件 -- 显示类
🔥 目录 [TOC]( 🔥 目录) 1. 前言 2. 显示类控件2.1 Label 1、显示不同文本2、显示图片3、文本对齐、自动换行、缩进、边距4、设置伙伴 3.2 LCD Number 3.3 ProgressBar 3.4 Calendar Widget 3. 共勉 🔥 1. 前言 之前我在上一篇文章【QT】…...

反馈驱动、上下文学习、多语言检索增强等 | Big Model Weekly 第55期
点击蓝字 关注我们 AI TIME欢迎每一位AI爱好者的加入! 01 A Bayesian Approach to Harnessing the Power of LLMs in Authorship Attribution 传统方法严重依赖手动特征,无法捕捉长距离相关性,限制了其有效性。最近的研究利用预训练语言模型的…...

CF 41A.Translation(Java实现)
题目分析 根据示例千言万语一句话,reverse 思路分析 将读取的值分ab,再将b.reverse和a比较,一样就YES 代码 import java.util.*;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);String …...

14【学历和能力哪个更重要】
这是很多学习的人有的一个疑问,并提出想让我发表下看法,前面一直没空,我刚好完结了一个项目,最近又有时间更新图文课程了,就展开来讲讲 主流的说法有2个 1:学历重要,依据是很多公司招聘都有学历…...

Learning Vue 读书笔记 Chapter 2
2. Vue 基本工作原理 2.1 Virtual DOM 概念: DOM: DOM以内存中树状数据结构的形式,代表了网页上的HTML(或XML)文档内容。它充当了一个编程接口,将网页与实际的编程代码(如JavaScript)连接起来…...

SpringBoot支持动态更新配置文件参数
前言 博主介绍:✌目前全网粉丝3W,csdn博客专家、Java领域优质创作者,博客之星、阿里云平台优质作者、专注于Java后端技术领域。 涵盖技术内容:Java后端、大数据、算法、分布式微服务、中间件、前端、运维等。 博主所有博客文件…...

开发技巧,vue 中的动态组件的引用 component + is
在项目中很多时候有切换 tab 的场景,一般来说都是用 v-if 或者 v-show 然后根据各种条件来控制显示隐藏。 其实我们可以使用 vue 中的动态组件,也能实现这个效果 <!-- currentTab 改变时组件也改变 --> <component :is"currentTab"…...

基于SpringBoot+WebSocket的前后端连接,并接入文心一言大模型API
前言: 本片博客只讲述了操作的大致流程,具体实现步骤并不标准,请以参考为准。 本文前提:熟悉使用webSocket 如果大家还不了解什么是WebSocket,可以参考我的这篇博客: rWebSocket 详解:全双工…...

PSD是什么图像格式?如何把PSD转为JPG格式?
在图形设计的世界里,Photoshop 文档(PSD)格式是 Adobe Photoshop 的原生文件格式,它允许设计师保存图像中的图层、蒙版、透明度和不同色彩模式等信息。对于需要进一步编辑的设计作品来说,PSD 文件提供了极大的灵活性。…...

c语言中mysql_query的概念和使用案例
在 C 语言中,使用 MySQL 数据库需要用到 MySQL C API。mysql_query() 函数是 MySQL C API 中的一个函数,用于执行 SQL 语句。 概念 mysql_query() 函数的原型如下: int mysql_query(MYSQL *mysql, const char *stmt_str)mysql:…...

一次端口监听正常,tcpdump无法监听到指定端口报文问题分析
tcpdump命令: sudo tcpdump -i ens2f0 port 6471 -XXnnvvv 下面是各个部分的详细解释: 1.tcpdump: 这是用于捕获和分析网络数据包的命令行工具。 2.-i ens2f0: 指定监听的网络接口。ens2f0 表示本地网卡),即计算机该指定网络接口捕…...

解决InnoDB: Failing assertion: !lock->recursive
背景: 在arm服务器里运行MySQL5.7.22版本 报错信息 : 2024-11-25T08:07:36.24182508:00 856 [Note] Multi-threaded slave statistics for channel : seconds elapsed 126; events assigned 53431297; worker queues filled over overrun level 0; …...

基于微信小程序的外卖点餐系统设计与实现ssm+论文源码调试讲解
4系统概要设计 4.1概述 本系统后台采用B/S结构(Browser/Server,浏览器/服务器结构)和基于Web服务两种模式,是一个适用于Internet环境下的模型结构。只要用户能连上Internet,便可以在任何时间、任何地点使用。系统工作原理图如图4-1所示: 图4-1系统工作原…...

Helm Chart 实现 Kubernetes 应用的多环境部署与镜像更新
在现代软件开发中,通常需要将应用部署到多个环境(如开发环境、测试环境、生产环境),并且在不同环境中使用不同的配置和镜像版本。Helm Chart 提供了强大的模板化和参数化功能,可以轻松实现多环境部署和镜像更新。本文将详细介绍如何使用 Helm Chart 实现 Kubernetes 应用的…...

“腾讯、钉钉、飞书” 会议开源平替,免费功能强大
在数字化时代,远程办公和线上协作越来越火。然而,市面上的视频会议工具要么贵得离谱,要么功能受限,甚至还有些在数据安全和隐私保护上让人不放心。 今天开源君给大家安利一个超棒的开源项目 - Jitsi Meet,这可是我在网…...

我谈区域偏心率
偏心率的数学定义 禹晶、肖创柏、廖庆敏《数字图像处理(面向新工科的电工电子信息基础课程系列教材)》P312 区域的拟合椭圆看这里。 Rafael Gonzalez的二阶中心矩的表达不说人话。 我认为半长轴和半短轴不等于特征值,而是特征值的根号。…...