线程安全-3 JMM

2024-05-28 01:28
文章标签 线程 安全 jmm

本文主要是介绍线程安全-3 JMM,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.谈一下JMM

1.JMM,JavaMemoryModel,Java内存模型。定义了多线程对共享内存读写操作的行为规范,通过规范多线程对共享内存的读写操作,以保证指令执行和结果的正确性。

2.JMM把内存分为两块

(1)主内存:是线程间共享的内存区域,可以被所有线程访问,存储了共享变量的原始副本。

(2)工作内存:是线程的私有区域,每个线程都有一个自己的工作内存,是线程的工作区域,不同线程的工作内存相互独立、相互隔离。

        a.线程的工作内存相互隔离,每个线程都只能访问属于自己的工作内存。

        b.线程不直接操作主内存的数据,而是将主内存的数据拷贝一份到自己的工作内存中,进行操作,操作完再将数据更新到主内存。

        c.线程在自己的工作内存对主内存变量的副本进行修改后,通过CAS操作将其更新到主内存的变量中,其他线程再将主内存变量的最新值更新到自己的副本变量中。不同线程是通过主内存进行交互的。

3.JMM的8个原子操作

(1)read:读取,读取主内存的变量值到工作内存中

(2)load:载入,将从主内存读到的变量值放到工作内存的副本变量中

(3)store:存储,将工作内存的变量值送到主内存中

(4)write:写入,将工作内存送来的变量值写入到主内存的变量中

(5)use:使用,将工作内存的变量值传递给执行引擎,以供其他指令需要

(6)assign:赋值,将从执行引擎获取到的值赋值给工作内存的变量

(7)lock:加锁,将主内存的变量标记为线程独占状态

(8)unlock:解锁,将主内存的变量的加锁状态解除

二.并发编程的三大特征是什么? / 导致并发程序出现问题的根本原因是什么?

1.原子性:一个操作要么全部完成,要么全部都不完成,不会因为上下文切换而导致结果出错。

2.可见性:虽然每个线程只能操作自己工作内存的数据,自己的工作内存对其他线程不可见;但是当一个线程对主内存的共享变量进行更新后,其他线程要能立即知道并更新为最新值。

3.有序性:为了提高运行效率,编译器会对代码进行重排序,cpu也会对指令进行重排序;这种重排序不会影响单线程的执行结果,但会影响多线程并发执行的结果。因此并发编程要能保证重排序之后的有序性,执行结果不会因重排序而出错。

三.JMM如何保证并发编程三大特征? / Java程序中如何保证多线程的执行安全?

1.原子性:synchronized、JUC中的Lock

2.可见性:volatile、synchronized、JUC中的Lock

3.有序性:volatile、synchronized

四.说一下volatile关键字

volatile关键字用于修饰共享变量(类的成员变量和静态成员变量),具有两种作用

1.保证并发编程的可见性

(1)问题1:JVM提供了一个即时编译器JIT,会对代码进行优化。例如while(!stop),stop默认为false。如果当前代码的执行逻辑中没有对stop进行修改,则会将代码优化为while(true),这在单线程下是可行的;但是在多线程中,若有其他线程对stop进行更改,由于代码优化,会导致执行当前代码的线程无法收到其他线程对stop的更改通知,失去了并发编程的可见性。

(2)问题2:若线程在更新主内存的共享变量后,其他线程未及时同步最新值,则其他线程在工作内存中的变量副本就相当于失效了,这也失去了并发编程的可见性。

(3)解决:使用volatile修饰共享变量

        a.使用volatile修饰的变量,可以防止JIT对其进行优化

        b.使用volatile修饰的变量,会对其读写操作加上属于硬件层面的内存屏障

                对volatile变量执行读操作前,会插入即读屏障,强行使当前工作内存的变量失效,重新去主内存获取变量值

                对volatile变量进行写操作后,会插入即写屏障,强行将工作内存的变量最新值更新到主内存中

2.保证并发编程的有序性

(1)问题:编译器为了提高效率会对代码进行重排序,影响了高并发下的执行结果

(2)解决:用volatile修饰的变量,会对其读写操作加上属于JMM层面的内存屏障,保证重排序后的有序性。

        LoadBarrier;

        volatile读操作; //重排序时,其上所有读操作不能越过屏障排到下面,其下所有写操作不能越过屏障跑到上面

        StoreBarrier;

        ...

        StoreBarrier;

        volatile写操作; //重排序时,其上所有写操作不能越过屏障排到下面,其下所有读操作不能越过屏障跑到上面

        LoadBarrier;

                

这篇关于线程安全-3 JMM的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1009151

相关文章

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

Java JUC并发集合详解之线程安全容器完全攻略

《JavaJUC并发集合详解之线程安全容器完全攻略》Java通过java.util.concurrent(JUC)包提供了一整套线程安全的并发容器,它们不仅是简单的同步包装,更是基于精妙并发算法构建... 目录一、为什么需要JUC并发集合?二、核心并发集合分类与详解三、选型指南:如何选择合适的并发容器?在多

Java中如何正确的停掉线程

《Java中如何正确的停掉线程》Java通过interrupt()通知线程停止而非强制,确保线程自主处理中断,避免数据损坏,线程池的shutdown()等待任务完成,shutdownNow()强制中断... 目录为什么不强制停止为什么 Java 不提供强制停止线程的能力呢?如何用interrupt停止线程s

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

SpringBoot实现虚拟线程的方案

《SpringBoot实现虚拟线程的方案》Java19引入虚拟线程,本文就来介绍一下SpringBoot实现虚拟线程的方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录什么是虚拟线程虚拟线程和普通线程的区别SpringBoot使用虚拟线程配置@Async性能对比H

在Java中实现线程之间的数据共享的几种方式总结

《在Java中实现线程之间的数据共享的几种方式总结》在Java中实现线程间数据共享是并发编程的核心需求,但需要谨慎处理同步问题以避免竞态条件,本文通过代码示例给大家介绍了几种主要实现方式及其最佳实践,... 目录1. 共享变量与同步机制2. 轻量级通信机制3. 线程安全容器4. 线程局部变量(ThreadL

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

Java中的xxl-job调度器线程池工作机制

《Java中的xxl-job调度器线程池工作机制》xxl-job通过快慢线程池分离短时与长时任务,动态降级超时任务至慢池,结合异步触发和资源隔离机制,提升高频调度的性能与稳定性,支撑高并发场景下的可靠... 目录⚙️ 一、调度器线程池的核心设计 二、线程池的工作流程 三、线程池配置参数与优化 四、总结:线程