Java中volatile关键字的作用是什么

Java中volatile关键字的作用是什么,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

创新互联建站主营芦淞网站建设的网络公司,主营网站建设方案,App定制开发,芦淞h5小程序开发搭建,芦淞网站营销推广欢迎芦淞等地区企业咨询

Jav内存模型(JMM)

谈到多线程就应该了解一下Java内存模型(JMM)的抽象示意图.下图:

Java中volatile关键字的作用是什么

线程A和线程B执行的是时候,会去读取共享变量(临界区),然后各自拷贝一份回到自己的本地内存,执行后续操作。
JMM模型是一种规范,就像Java的接口一样。JMM会涉及到三个问题:原子性,可见性,有序性。
所谓原子性。就是说一个线程的执行会不会被其他线程影响的。他是不可中断的。举个例子:

int i=1

这个语句在Jmm中就是原子性的。无论是一个线程执行还是多个线程执行这个语句,读出来的i就是等于1。那什么是非原子性呢,按道理如果Java的代码都是原子性,应该就不会有线程问题了啊。其实JMM这是规定某些语句是原子性罢了。举个非原子性例子:

i ++;

这个操作就不是原子性的了。因为他就是包含了三个操作:第一读取i的值,第二将i加上1,第三将结果赋值回来给i,更新i的值。
所谓可见性。可见性表示如果一个值在线程A修改了,线程B就会马上知道这个结果。

所谓有序性。所谓有序性值的是语意的有序性。就是说代码顺序可能会发生变化。因为有一个指令重排机制。所谓指令重排,他会改变代码执行顺序,为了让cpu执行效率更高。为了防止重排序出错,JMM有个happen-before规则,这个规则限制了那些语句执行在前,那些语句执行在后。

Happen-before:

程序顺序原则:一个线程内保证语义的串行性

volatile原则:volatile变量的写发生在读之前

锁规则:先加锁再解锁

传递性:a先于b,b先于c,则a必定先于c

线程的start方法先于他的每一个操作

线程所有的操作先于线程的终结

对象的构造函数执行、结束先于finalize()方法。

volatile

进入正题,volatile可以保证变量(临界区)的可见性以及有序性,但是不能保证原子性。举个例子:

public class VolatileTest implements Runnable{
 private static VolatileTest volatileTest = new VolatileTest();
 private static volatile int i= 0;
 public static void main(String[] args) throws InterruptedException {
  for (int j = 0; j < 20; j++) {
   Thread a = new Thread(new VolatileTest());
   Thread b = new Thread(new VolatileTest());
   a.start();b.start();
   a.join();b.join();
   System.out.print(i+"&&");
  }

 }
 
 @Override
 public void run() {
  for (int j = 0; j < 1000; j++) {
   i++;
  }
 }

}

// 输出结果
// 2000&&4000&&5852&&7852&&9852&&11852&&13655&&15655&&17655&&19655&&21306  
//&&22566&&24566&&26189&&28189&&30189&&32189&&34189&&36189&&38089&&

有结果看到有问题,虽然i已经添加了volatile关键字,说明volatile关键字不能保证i++的原子性。

那什么场景适合使用volatile关键字

1、轻量级的“读-写锁”策略

private volatile int value;
public int getValue(){ return value;}
public synchronized void doubleValue(){ value = value*value; }

2.单例模式(双检查锁机制

private volatile static Singleton instace; 
public static Singleton getInstance(){ // 没有使用同步方法,而是同步方法块
 //第一次null检查 ,利用volatile的线程间可见性,不需要加锁,性能提高 
 if(instance == null){   
  synchronized(Singleton.class) { //锁住类对象,阻塞其他线程
   //第二次null检查,以保证不会创建重复的实例  
   if(instance == null){  
    instance = new Singleton(); // 禁止重排序
   } 
  }   
 } 
 return instance;

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注创新互联行业资讯频道,感谢您对创新互联的支持。

网站题目:Java中volatile关键字的作用是什么
本文来源:https://www.cdcxhl.com/article32/jhhgsc.html

成都网站建设公司_创新互联,为您提供网站设计自适应网站网站收录关键词优化网站策划网站内链

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联

绵阳服务器托管