JMM 简单理解
JMM 简单理解
1 Java 内存模型
Java 内存模型(Java Memory Model,JMM),主要为了屏蔽各种硬件和操作系统的内存差异,以实现让 Java 程序在各种平台下都能达到一致的内存访问效果,而设计的
2 工作内存与主内存
JMM 包括两大区域,工作内存和主内存。下图表示线程、工作内存、主内存三者的交互关系
线程对应的的工作内存,保存该线程所使用的主内存变量的副本,线程对变量的所有操作(读取赋值等)都必须在工作内存中进行,而不能直接读写主内存中的数据,不同线程也无法访问对方工作内存中的变量
3 内存间的交互操作
- lock(锁定):作用于主内存的变量,它把一个变量标识为一条线程独占的状态
- unlock(解锁):作用于主内存的变量,它把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定
- read(读取):作用于主内存的变量,它把一个变量的值从主内存传输到线程的工作内存中,以便随后的 load 动作使用
- load(载入):作用于工作内存的变量,它把 read 操作从主内存中得到的变量值放入工作内存的变量副本中
- use(使用):作用于工作内存的变量,它把工作内存中一个变量的值传递给执行引擎,每当虚拟机遇到一个需要使用变量的值的字节码指令时将会执行这个操作
- assign(赋值):作用于工作内存的变量,它把一个从执行引擎接收的值赋给工作内存的变量, 每当虚拟机遇到一个给变量赋值的字节码指令时执行这个操作
- store(存储):作用于工作内存的变量,它把工作内存中一个变量的值传送到主内存中,以便随后的 write 操作使用
- write(写入):作用于主内存的变量,它把 store 操作从工作内存中得到的变量的值放入主内存的变量中
4 经典代码案例
package com.fzshuai.volatile1;import java.util.concurrent.TimeUnit;/*** @author fzshuai* @date 2023/10/26 23:32* @since 1.0*/
public class JMMDemo {private static int num = 0;public static void main(String[] args) {new Thread(() -> {while (num == 0) {}}).start();try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {throw new RuntimeException(e);}num = 1;System.out.println(num);}
}
为什么程序一直在执行没有停止呢?让我们来看看线程与内存之间的交互,不难发现线程 B 改变了变量 num 的值,但是线程 A 并不知道