Java Memory Model
我们常说的JVM内存模式指的是JVM的内存分区;而Java内存模型是一种虚拟机规范。
JMM规范用于屏蔽掉各种硬件和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的并发效果
JMM规范了Java虚拟机与计算机内存是如何协同工作的,规定了一个线程何时可以看到由其他线程修改过后的共享变量的值。
JMM三大特性
原子性
什么是原子性?
原子性是指所有的操作是不可分的,要么全部一起执行,要么都不执行。
什么情况会导致原子性问题?
线程切换
java如何保证原子性?
java中实现原子操作的方法大致有2种:加锁
可见性
什么是可见性?
可见性是指一个线程对共享变量的修改,对于另一个线程是可以看到的。
为什么会导致可见性问题?
JMM有如下规定:
我们定义的所有变量都储存在 主内存中
每个线程都有自己 独立的工作内存,里面保存对主内存中该变量的一份拷贝
为什么这样处理?
为了弥补CPU与内存
之间的速度差距,不同线程将主存的数据拷贝到CPU缓存即工作内存
,读取写入都操作CPU缓存,然后将缓存中的数据写入主存中,这样一来就解决了速度差距。
线程对共享变量所有的操作都必须在自己的工作内存中进行,不能直接从主内存中读写(不能越级访问)
不同线程之间也无法直接访问其他线程的工作内存中的变量,线程间变量值的传递需要通过主内存来进行。(同级不能相互访问)
所以a线程修改的内存不能及时被b线程看到就会导致可见性问题
java保证可见性的方式有哪些?
java提供了volatile来实现
有序性
有序性指的是程序按照代码的先后顺序执行。
为什么会导致有序性问题?
指令重排序
为了性能优化,编译器会进行指令重排序,改变程序语句的先后顺序。
为什么指令重排序可以提高性能?
简单地说就是指令1还没有执行完,就可以开始执行指令2,而不用等到指令1执行结束之后再执行指令2,这样就大大提高了效率。
指令重排序导致问题举例:
在单例模式的实现上有一种双重检验锁定的方式,代码如下:
public class DoubleCheck {
private static volatile DoubleCheck instance=null;
private DoubleCheck(){}
public static DoubleCheck getInstance(){
if(instance==null){ //这里判空是为了提升性能
synchronized (DoubleCheck.class){
if(instance==null){//这里判空是必须的
instance=new DoubleCheck();
}
}
}
return instance;
}
}
我们先看 instance=new DoubleCheck();
未被编译器优化的操作:
指令1:分配一款内存M
指令2:在内存M上初始化DoubleCheck对象
指令3:将M的地址赋值给instance变量
编译器优化后的操作指令:
指令1:分配一块内存S
指令2:将M的地址赋值给instance变量
指令3:在内存M上初始化DoubleCheck对象
现在有2个线程,刚好执行的代码被编译器优化过,过程如下:
最终线程B获取的instance是没有初始化的,此时去使用instance可能会产生一些意想不到的错误。
那java如何禁止指令重排序保证有序性?
java提供了volatile、synchronized、Lock锁来实现