随着互联的飞速发展,互联网公司也越来越高,并发多线程,内存管理,JVM调优等成为面试必问题。
1. 什么是 volatile?
volatile 是 Java 虚拟机提供的轻量级的同步机制。
他的三大特性:
- 保证可见性
- 不保证原子性
- 禁止指令重排
1.1 可见性
在解释可见性之前需要先看一下什么是 JMM?
JMM(Java 内存模型 Java Memory Model,简称 JMM)本身是一种抽象的概念,并不真实存在,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(实例字段,静态字段和构成数组对象的元素)的访问方式。
JMM 关于同步的规定:
- 线程解锁前,必须把共享变量的值刷新回主内存
- 线程加锁前,必须读取主内存的最新值到自己的工作内存
- 加锁解锁是同一把锁
JMM 保证:可见性、原子性、有序性
由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存(有些地方成为栈空间),工作内存是每个线程的私有数据区域,而Java内存模型中规定所有变量都存储在主内存,主内存是共享内存区域,所有线程都可访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先要将变量从主内存拷贝到自己的工作空间,然后对变量进行操作,操作完成再将变量写回主内存,不能直接操作主内存中的变量,各个线程中的工作内存储存着主内存中的变量副本拷贝,因此不同的线程无法访问对方的工作内存,此案成间的通讯(传值) 必须通过主内存来完成,其简要访问过程如下图:
可见性:当一个线程修改主物理内存的值再写回主内存时,其他线程能都收到修改数据的通知。
如果看不懂上面的一段话,那么看看代码,然后再会过头看上面的文字。代码演示:
class MyData{
//第一次不加 ,第二次运行加 volatile
int n = 0;
public void change() {
this.n=60;
}
}
/**
* 1.验证 volatile 的