其他
你真的了解 volatile 吗?
点击上方 Java后端,选择 设为星标
优质文章,及时送达很早就接触了volatile,但是并没有特别深入的去研究它,只有一个朦胧的概念,就是觉得用它来解决可见性的,但可见性又是什么呢?
最近经过查阅各种资料,并结合自己的思考和实践,对volatile有了比较深刻的认识,在此总结并分享给大家。
可见性
如何理解可见性,还是来看个会出现死循环的例子:
(注意:运行时请加上jvm参数:-server,while循环内不要有标准输出):
这是为什么呢?先来看看java的内存模型,如下图:
java内存分为工作内存和主存
工作内存:即java线程的本地内存,是单独给某个线程分配的,存储局部变量等,同时也会复制主存的共享变量作为本地的副本,目的是为了减少和主存通信的频率,提高效率。
主存:存储类成员变量等
可见性是指的是线程访问变量是否是最新值。
局部变量不存在可见性问题,而共享内存就会有可见性问题,因为本地线程在创建的时候,会从主存中读取一个共享变量的副本,且修改也是修改副本,且并不是立即刷新到主存中去,那么其他线程并不会马上共享变量的修改。
因此,线程B修改共享变量后,线程A并不会马上知晓,就会出现上述死循环的问题。
解决共享变量可见性问题,需要用volatile关键字修饰。
如下图代码就不会出现死循环:
那么为什么能解决死循环的问题呢?
可见性的特性总结为以下2点:
对volatile变量的写会立即刷新到主存 对volatile变量的读会读主存中的新值
问题1:t2时刻,如果线程A读取running变量,会读取到false,还是等待线程B执行完呢?
问题2:t4时刻,线程A是否一定能读取到线程B修改后的最新值
同步块存在如下语义:
进入同步块,访问共享变量会去读取主存 退出同步块,本地内存对共享变量的修改会立即刷新到主存
volatile变量的原子性
对一个volatile变量的写操作,只有所有步骤完成,才能被其它线程读取到。 多个线程对volatile变量的写操作本质上是有先后顺序的。也就是说并发写没有问题。
//线程1初始化User
User user;
user = new User();
//线程2读取user
if(user!=null){
user.getName();
}
分配对象的内存空间 初始化对线 设置user指向刚分配的内存地址
1->3->2
对volatile理解的误区
这其实是不对的。
然而运行的结果是:11349
读取i=0 计算i+1=1,并重新赋值给i
i++这种操作不是原子操作 volatile 并不会有锁的特性