了解JMM需要我们去知道什么是Volatile。
谈谈你对Volatile的理解:
1.保证可见性
2.不保证原子性
3.禁止指令重排
那么什么是JMM呢?其实JMM只是一个概念,并不是显示存在的东西。
JMM:Java内存模型
关于JMM的同步约定:
1.线程解锁前需要共享变量立刻刷到主存中。
2.线程加锁前必须读取主存中的最新值到工作内存中。
3.加锁和解锁必须是一把锁。
内存划分
JMM的内存主要划分为主内存和工作内存
内存相关操作
内存交互操作有8种,虚拟机实现必须保证每一个操作都是原子的,不可在分的(对于double和long类型的变量来说,load、store、read和write操作在某些平台上允许例外)
- lock (锁定):作用于主内存的变量,把一个变量标识为线程独占状态
- unlock (解锁):作用于主内存的变量,它把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定
- read (读取):作用于主内存变量,它把一个变量的值从主内存传输到线程的工作内存中,以便随后的load动作使用
- load (载入):作用于工作内存的变量,它把read操作从主存中变量放入工作内存中
- use (使用):作用于工作内存中的变量,它把工作内存中的变量传输给执行引擎,每当虚拟机遇到一个需要使用到变量的值,就会使用到这个指令
- assign (赋值):作用于工作内存中的变量,它把一个从执行引擎中接受到的值放入工作内存的变量副本中
- store (存储):作用于主内存中的变量,它把一个从工作内存中一个变量的值传送到主内存中,以便后续的write使用
- write (写入):作用于主内存中的变量,它把store操作从工作内存中得到的变量的值放入主内存的变量中
JMM对这八种指令的使用,制定了如下规则:
- 不允许read和load、store和write操作之一单独出现。即使用了read必须load,使用了store必须write
- 不允许线程丢弃他最近的assign操作,即工作变量的数据改变了之后,必须告知主存
- 不允许一个线程将没有assign的数据从工作内存同步回主内存
- 一个新的变量必须在主内存中诞生,不允许工作内存直接使用一个未被初始化的变量。就是怼变量实施use、store操作之前,必须经过assign和load操作
- 一个变量同一时间只有一个线程能对其进行lock。多次lock后,必须执行相同次数的unlock才能解锁
- 如果对一个变量进行lock操作,会清空所有工作内存中此变量的值,在执行引擎使用这个变量前,必须重新load或assign操作初始化变量的值
- 如果一个变量没有被lock,就不能对其进行unlock操作。也不能unlock一个被其他线程锁住的变量
- 对一个变量进行unlock操作之前,必须把此变量同步回主内存
刚刚谈到了Volatile,Volatile不保证原子性,那么我们怎么才能让它保证原子性呢?
方法一:加锁 Synchronized或者Lock锁。
方法二:JDK为我们提供了一个保证原子性的包:
‘使用了Volatile
package com.ys.VTest;
public class VaTest {
private static volatile int num = 0;
public static void add(){
num++;
}
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
new Thread(()->{
for (int i1 = 0; i1 < 100; i1++) {
add();
}
}).start();
}
while (Thread.activeCount()>2){
Thread.yield(); //线程礼让
}
System.out.println(num);
}
}
结果是不保证原子性的,每次的结果都不经相同,那么怎么解决呢?
在add方法上锁就能解决(以Synchronized为例):
package com.ys.VTest;
public class VaTest {
private static volatile int num = 0;
public synchronized static void add(){
num++;
}
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
new Thread(()->{
for (int i1 = 0; i1 < 100; i1++) {
add();
}
}).start();
}
while (Thread.activeCount()>2){
Thread.yield();
}
System.out.println(num);
}
}
结果:
现在有个问题就是我们要保证原子性但是不用加锁的方式去解决,应该怎么办?
使用原子类。
对以上代码修改:
package com.ys.VTest;
import java.util.concurrent.atomic.AtomicLong;
public class VaTest {
private static volatile AtomicLong num = new AtomicLong();
public static void add(){
num.getAndIncrement();//加1
}
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
new Thread(()->{
for (int i1 = 0; i1 < 100; i1++) {
add();
}
}).start();
}
while (Thread.activeCount()>2){
Thread.yield();
}
System.out.println(num);
}
}
原子类的包装类是CAS,原子类比较高效,比锁还要高效。
以上是不保证原子性的,接下来验证下保证可见性。
package com.ys.VTest;
import java.util.concurrent.TimeUnit;
public class Demo2 {
private static int a =0;
public static void main(String[] args) {
new Thread(()->{
while (a == 0){
}
}).start();
try {
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
a++;
System.out.println(a);
}
}
这时候代码是不会停止的,因为线程在死循环。
Volatile就可以去让这个线程停止,具有可见性。
package com.ys.VTest;
import java.util.concurrent.TimeUnit;
public class Demo2 {
private volatile static int a =0;
public static void main(String[] args) {
new Thread(()->{
while (a == 0){
}
}).start();
try {
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
a++;
System.out.println(a);
}
}
指令重排
指令重排是指计算机并不会完全按照我们的代码顺序执行,处理器在进行指令重排的时候是考虑到数据之间的依赖性的,不会去乱排导致代码执行不下去。 Volatile可以去解决这个问题