Java并发中的原子性、可见性和有序性

news2025/1/11 9:57:31

基于JMM的内存模型,Java并发编程的核心问题:原子性、可见性、有序性

那么在此之前,我们有必要先说一下Java的JMM内存模型:java内存模型,是java虚拟机规范的一种工作模式,它将内存分为主内存和工作内存。线程在操作变量时,会将内存中的数据复制一份到工作内存,在工作内存中操作完成后,在写回到主内存中。

Java内存模型干了什么?

它定义了线程之间如何通过内存进行通信以及如何进行同步来保证程序正确性;Java内存模型规定了一个线程读取到的数据是另一个线程写入的最新版本。因为在并发程序中多个线程可以同时访问共享的变量,如果不采取措施来避免线程并发访问导致的意外结果,程序可能会出现各种未知的问题,如数据不一致、死锁等,而Java内存模型就是为了解决这些问题而设计的。

变量数据存储在主内存中,线程在操作变量时,会将主内存中的数据复制一份到工作内存,在工作内存中操作完成后,在写回到主内存中。

注:这里的本地内存只是JMM的抽象,是一个虚拟的概念,并不是真实存在一块内存叫本地内存,它是用来描述Java线程之间的可见性、有序性和原子性法则的一种规范。

1.原子性

原子性(Atomic)就是不可分割的意思,是指在进行一系列操作的时候这些操作要么全部执行要么全部不执行,不存在只执行一部分的情况。

上面的一系列操作,具体指的是什么呢?

答案就是那些对共享变量的读写操作,其实这也涉及到了整个并发编程的核心:处理多线程情况下线程之间的竞争条件和对共享变量的访问。

我们都知道并发指的是微观上同一时间段内多个线程交替执行,宏观上是多个线程一起执行,那么还因为我们现在的cpu是多核的,并且基于JMM的内存模型,这种种的原因就导致我们总是会面对这样一种问题:在同一时间内有多个线程对同一个共享变量进行读写操作,怎样才能保持这些操作结果是正确的呢,其实就是保持这三个性质:原子性,可见性,有序性。如何保持这三个特性,我想这就是并发编程这个篇章要解决的问题了。

为什么要保持原子性?

因为对于多个线程同时访问同一个共享变量时,如果没有保证原子性,就有可能发生线程安全问题。

例如,多个线程同时对一个计数器变量进行累加操作,如果没有保证原子性,就有可能出现计数结果不正确的情况。为了避免这种情况发生,可以将计数器变量的类型定义为AtomicInteger,或者使用synchronized关键字对读写操作加锁,保证多个线程对计数器变量进行操作时,每次只有一个线程在进行操作,从而保证了原子性。


如何保持原子性?

1.加锁:使用Synchronized或ReentrantLock

我们对读写共享资源的部分代码加锁,一个线程获得锁后,另一个线程想访问这段代码时,就会线程阻塞,直到另一个线程释放锁;所以加锁是一种阻塞式实现,而且这是一种悲观锁思想的实现。

2.使用原子类进行并发操作

java中,还提供了一些java.util.concurrent包中的原子类;是一种无锁实现;在低并发情况下使用;采用了CAS机制(Compare-And-Swap)。

总之,原子类的原子性是通过 volatile + CAS 实现原子操作的。 如AtomicInteger类,AtomicInteger 类中的 value 是有 volatile 关键字修饰的,这就保证了 value的内存可见性,这为后续的 CAS 实现提供了基础。

CAS机制(比较重要,面试高频)

CAS机制(Compare-And-Swap):比较并交换,该算法是硬件对于并发操作的支持;

是乐观锁的一种实现方式;采用自旋的思想,是一种轻量级的锁机制(乐观锁是无锁实现)。

CAS全称是compare and swap 比较并且交换,是一种乐观锁思想,即无锁实现。该算法是硬件对于并发操作的支持;

建议在低并发情况下使用,

主要思想是自旋思想:第一次采取内存值v到工作内存中,进行计算后得到更新值B,然后进行判断,从主内存中再次取出V,记作A;当且仅当A==V时,我们才可以将B更新回主内存,否则,说明在之前已经有线程将改动了,我们就用这个改动后的值重复刚才的过程。

特点:

  • 是一种无锁实现;

  • 只能在低并发情况下使用;

  • 不加锁,所有线程都可以对共享数据操作;

  • 由于不加锁,所以不会阻塞,效率比加锁高;

  • 采用自旋思想;

CAS机制中的不断自旋(spinlock)是指当多个线程同时请求对同一个变量进行操作时,如果这个变量的值不符合线程的预期,那么线程会反复尝试重新取值进行计算直到成功为止,而不是放弃处理或者进入睡眠状态。

使用CAS机制的优点是没有锁的开销,由于不需要线程间上下文切换的开销,比使用锁机制性能要高。但是同时也可能会出现自旋等待增加的问题(比如线程数量过多),从而导致性能下降。因此,在使用CAS时需要进行适当的调优

 2.可见性

可见性是指一个线程修改的共享变量的值可以被其他线程及时地看到。当一个线程修改了共享变量的值而其他线程还在使用这个值时,如果其他线程看到的仍然是修改之前的旧值,就会导致程序出错或者运行出现异常。

如何保持可见性?

正确地保证可见性是并发编程的重要课题。为了解决可见性的问题,Java提供了多种解决方案:

  1. 加锁同步机制:通过synchronized关键字或ReentrantLock类等加锁同步机制,可以保证共享变量的读写操作具有原子性、可见性和有序性。

  2. volatile关键字:使用volatile关键字可以保证被修饰的变量对于所有线程都是可见的,并且对共享变量的读写操作具有原子性。

  3. final关键字:被final关键字修饰的变量可以保证在所有线程中都具有可见性,但是只适用于不可变的变量。

  4. Atomic变量:Java中的Atomic包提供了对基本数据类型的高效跨多线程操作,具有可见性和原子性的保证。

有了volatile保证了变量的改变是立即可见的,是不是就可以不用加锁了?

虽然volatile关键字能够保证变量的可见性和一定的有序性,但是并不等同于加锁的同步机制,不能用来替代加锁同步机制。

加锁同步机制(如使用synchronized方法或代码块、ReentrantLock类等)在保证可见性和有序性的同时,还能够保证多个线程对共享变量的操作是原子性的,从而避免了一些其他的线程安全问题,而使用volatile修饰共享变量只能保证可见性和一定的有序性,无法保证复合操作的原子性。

总之,虽然volatile关键字能够提供可见性和有序性的保证,但是它不能代替加锁同步机制,开发者需要根据实际情况选择合适的同步机制来确保线程安全。

3.有序性

有序性指的是程序按照代码的先后顺序执行

为了优化性能,有时候java会自动对一些代码指令的执行顺序调整重排,以提高速度。在某种情况下,顺序调整后,可能会对后续代码操作进行影响。

如何保证有序性?

使用volatile关键字修饰的变量,在执行的过程中与它相关的代码不会被重排序执行。可解决无序性问题

总结:volatile关键字就可以保证有序性和可见性,但是原子性的保证比较复杂,有加锁和不加锁两种方式保证原子性。我们对共享变量的并发操作只有保持了这三个性质,就可以保持操作的正确性,这就是java并发编程的真正目的!

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

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

相关文章

【机器学习】十大算法之一 “PCA”

作者主页:爱笑的男孩。的博客_CSDN博客-深度学习,活动,python领域博主爱笑的男孩。擅长深度学习,活动,python,等方面的知识,爱笑的男孩。关注算法,python,计算机视觉,图像处理,深度学习,pytorch,神经网络,opencv领域.https://blog.csdn.net/Code_and516?typeblog个…

高性能计算的意义是什么

高性能计算(High-Performance Computing,HPC)在现代科学、工程和商业领域中具有重要意义。以下是一些高性能计算的重要意义: 加速科学研究和创新:高性能计算可以提供大规模的计算能力和存储资源,使得科学家…

stm32数据对齐、PRESERVE8、freertos堆栈

为什么需要数据对齐? 避免数据在内存中跨边界存储,减少读取数据次数,提高效率,本质上是以空间换时间的做法 下图中属于同一水平位置的为同一边界 变量在同一边界里的一次存储周期就可以读取 一旦跨了上下两个边界来存储就需要至少…

干爆源码系列之Step by step lldb/gdb调试多线程

Step by step lldb/gdb调试多线程 0.叙谈1.断点分析2.多线程切换 2.1 并发队列 2.1.1 两次入队 2.2 线程调度 2.2.1 执行build端子MetaPipeline 2.2.1.1 Thread6调度第一个PipelineInitializeTask 2.2.1.2 Thread7调度第二个Pipelin…

TypeScript 数据联合类型的解读

概念: 联合类型(Union Types)表示取值可以为多种类型中的一种,或者也可以理解将多个类型合并为一个类型对变量进行注解。 语法结构: 联合类型使用 | 分隔每个类型。 let 变量:类型1 | 类型2 | 类型3… 案列…

基于Java校园代购服务订单系统设计实现(源码+lw+部署文档+讲解等)

博主介绍: ✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战 ✌ 🍅 文末获取源码联系 🍅 👇🏻 精…

你的前端技术是否能通过这些高频面试题?

文章目录 1.储存了某个数据到 localStorage,立即进行 localStorage.getItem,能否取到值?2.实现异步的方式有几种3.异步不阻塞4.选择 div 的第二个子元素5.display: none 和 visibility: hidden 的区别6.如果想要让一个元素的宽高包含元素的 b…

【初识C语言】选择语句+循环语句+函数+数组

文章目录 前言1. 选择语句2. 循环语句3. 函数4. 数组 前言 C语言是一门结构化的程序设计语言 顺序结构; 选择结构; 循环结构。 1. 选择语句 生活中处处面临着选择,如果你好好学习,校招时拿一个好offer,走上人生巅峰。…

关于程序员的工作总结

程序员工作总结篇1 从我x月x日进入公司到现在已经过去一年了,从一名刚刚结束实习的学生到一名独立的开发人员,角色改变了,职责也改变了。虽然已经预计了工作之中会有很多困难,可是在实际的项目开发中,自己所遇到远远不…

(超级详细)如何在Mac OS上的VScode中配置OpenGL环境并编译

文章目录 安装环境下载GLAD与GLFW一、下载GLAD二、下载GLFW 项目结构配置测试程序与项目的编译测试可执行文件HelloGL 安装环境 机器:macbook air 芯片: M1芯片(arm64) macOS:macOS Ventura 13.4 VScode version&#…

Pytorch数据类型Tensor张量操作(操作比较全)

文章目录 Pytorch数据类型Tensor张量操作一.创建张量的方式1.创建无初始化张量2.创建随机张量3.创建初值为指定数值的张量4.从数据创建张量5.生成等差数列张量 二.改变张量形状三.索引四.维度变换1.维度增加unsqueeze2.维度扩展expand3.维度减少squeeze4.维度扩展repeat 五.维度…

SpringCloud Alibaba入门5之使用OpenFegin调用服务

我们继续在上一章的基础上进行开发 SpringCloud Alibaba入门4之nacos注册中心管理_qinxun2008081的博客-CSDN博客 Feign是一种声明式、模板化的HTTP客户端。使用Feign,可以做到声明式调用。Feign是在RestTemplate和Ribbon的基础上进一步封装,使用RestT…

SAP从入门到放弃系列之BOM行项目-虚拟装配-Part4

文章目录 虚拟组件(Phantom assemblies):作用:BOM中虚拟件维护的方式: 物料主数据维度BOM组件维度(数据优先级最高) BOM组件的展开类型:BOM组件的特殊采购类数据测试示例&#xff1…

基于open62541库的OPC UA协议节点信息查询及多节点数值读写案例实践

目录 一、OPC UA协议简介 二、open62541库简介 三、 opcua协议的多点查询、多点读写案例服务端opcua_server 3.1 opcua_server工程目录 3.2 程序源码 3.3 工程组织文件 3.4 编译及启动 四、opcua协议的多点查询、多点读写案例客户端opcua_client 4.1 opcua_client工程目录 4…

医院管理系统源码PACS超声科室源码DICOM影像工作站

一、医学影像系统(PACS)是一种应用于医院影像科室的系统,主要任务是将日常产生的各种医学影像(如核磁、CT、超声、X光机、红外仪、显微仪等设备产生的图像)通过各种接口(模拟、DICOM、网络)以数…

社区活动 | OpenVINO™ DevCon 中国系列工作坊第二期 | 使用 OpenVINO™ 加速生成式 AI...

生成式 AI 领域一直在快速发展,许多潜在应用随之而来,这些应用可以从根本上改变人机交互与协作的未来。这一最新进展的一个例子是 GPT 模型的发布,它具有解决复杂问题的能力,比如通过医学和法律考试这种类似于人类的能力。然而&am…

Android Studio实现推箱子小游戏

项目目录 一、项目概述二、开发环境三、详细设计四、运行演示五、项目总结 一、项目概述 推箱子是一款非常受欢迎的益智游戏,游戏的玩法简单,但是需要玩家具备一定的逻辑思维能力和空间感知能力,因此深受广大玩家的喜爱。在游戏中&#xff0…

正点原子F4HAL库串口中断再解读

七步走,参考usart.c文件 void HAL_UART_MspInit(UART_HandleTypeDef *huart) 这个函数进行了(1)、(2)、(3)、(5)中的使能中断 void uart_init(u32 bound)函数进行了&…

『手撕 Mybatis 源码』07 - Proxy 对象创建

Proxy 对象创建 问题 sqlSession.getMapper(UserMapper.class) 是如何生成的代理对象? Mapper 代理方式初始化完成后,下一步进行获取代理对象来执行 public class MybatisTest {/*** 问题2:sqlSession.getMapper(UserMapper.class); 是如…

EMC学习笔记(五)传输线模型及反射、串扰

1.概述 在高速数字电路PCB设计中,当布线长度大于20分之一波长或信号延时超过6分之一信号上升沿时,PCB布线可被视为传输线。传输线有两种类型:微带线和带状线。与EMC设计有关的传输线特性包括:特征阻抗、传输延迟、固有电容和固有电感。反射与串扰会影响…