JMM模型
首先你要有JMM(java memory model)的相关知识
如果没有请看这篇文章:JMM和JVM内存模型
volatile定义
volatile是java虚拟机提供的轻量级的同步机制
三个特性:1. 保证可见性 2.不保证原子性 3.禁止指令重排
保证可见性
验证代码
public class VolatileDemo {
boolean running = true;
void m(){
System.out.println("m start");
while (running) {
}
System.out.println("m end!");
}
public static void main(String[] args){
VolatileDemo volatileDemo = new VolatileDemo();
new Thread(volatileDemo::m,"t1").start();
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
volatileDemo.running = false;
}
}
不保证原子性
验证代码
class MyData{
volatile int number = 0;
public void add(){
number++;
}
}
public class VolatileDemo2 {
public static void main(String[] args){
MyData myData = new MyData();
for (int i = 0; i < 20; i++) {
new Thread(() ->{
for (int j = 0; j < 1000; j++) {
myData.add();
}
}).start();
}
while (Thread.activeCount()>2) {
Thread.yield();
}
System.out.println(Thread.currentThread().getName()+"finally number value:"+myData.number);
}
}
如何解决
1、加入synchronized
2、使用juc下的AtomicInteger
禁止指令重排(有序性)
指令重排:多线程环境中线程交替执行,由于编译器优化重排的存在,两个线程中使用的变量能否保证一致性是无法确定的,结果无法预测。
指令重排过程:
源代码 -> 编辑器优化的重排 -> 指令并行的重排 -> 内存系统的重排 ->最终执行的指令
但是加了volatile的关键字之后就禁止电脑进行指令重排,从而避免多线程环境下程序出现乱序执行的现象