这篇文章给大家介绍如何在Java中使用volatile关键字,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。
volatile 关键字:当多个线程进行操作共享数据时,可以保证内存中的数据可见。 相较于 synchronized 是一种较为轻量级的同步策略。
缺点:
1. volatile 不具备“互斥性”
2. volatile 不能保证变量的“原子性”
很多资料中是这样介绍volatile关键字的:
volatile是轻量级的synchronized,它在多处理器开发中保证了共享变量的“可见性”。可见性的意思是当一个线程修改一个共享变量时,另外一个线程能读到这个修改的值。
文字不太好理解,通过例子来理解。
1、例子
首先看一个没有使用volatile关键字例子:
package com.swnote.java; /** * volatile测试例子 * * @author lzj * @date [2019-04-47] */ public class VolatileTest { private boolean flag; public static void main(String[] args) { VolatileTest test = new VolatileTest(); test.test(); } public void test() { new Thread(() -> { try { Thread.sleep(1000L); } catch (InterruptedException e) { e.printStackTrace(); } flag = true; }).start(); new Thread(() -> { while (true) { if (flag) { System.out.println("thread flag = " + flag); } } }).start(); } }
该例子中定义了一个flag共享变量,test方法里面开启了两个线程,第一个线程在等待1秒中后修改共享变量flag的值为true,第二个线程通过循环判断flag的值,当flag的值为true时,输出内容。
此时有两种猜想:
执行后,可以看到输出内容,即说明第二个线程能够感知到第一个线程对共享变量flag的修改
执行后,没有任务内容,即说明第二个线程无法感知到第一个线程对共享变量flag的修改
然后执行结果为:
没有任务的输出内容,即证明了此时这样情况下第二个线程无法感知到第一个线程对共享变量flag的修改的
现在修改一下例子,即为flag变量加上volatile关键字,即:
private volatile boolean flag;
然后再运行,此时结果为:
此时就有内容输出了,说明加上volatile关键字后,第二个线程可以感知到第一个线程对共享变量flag的修改的,这就是上面概念中所说的volatile在多处理器开发中保证了共享变量的“可见性”。
2、原理
通过上面的例子证明了volatile在多处理器开发中保证了共享变量的“可见性”,那它是怎么实现的呢?
这时就得介绍一下Java的内存模型了,首先看如下示意图:
Java内存模型是如上面所示的:
共享变量存储在主内存中,每个线程都有一个私有的本地内存,本地内存保存了被该线程使用到的主内存的副本拷贝,线程对变量的所有操作都必须在自己的本地内存中进行,而不能直接读写主内存中的变量。
根据此理解,上述例子的在没有加volatile时的情况是这样的:
第一个线程从主内存中获取共享变量flag的值,此时值为false,将该值放到自己的本地内存中,然后对变量进行修改,将值改为true,此时也只是将本地内存中flag的值改为了true,此时还没有将值同步到主内存中,然后第二线程也是将共享变量flag的值放到自己的本地内存中,而此时flag的值还是为false,所以就是一直没有内容输出了。
然而加上volatile关键字后,第一个线程对flag的修改会强制刷新到主内存中去,同时还会导致其他线程中的本地内存的值会无效,需要重新到主内存获取,这样就保证了第一个线程对flag修改后,第二线程能够感知到。
3、注意点
volatile是轻量级的synchronized,但是它是不能够代替synchronized的,因为volatile只能保证原子性操作的安全,对于复合操作,volatile是不能保证线程安全的。
例如:
package com.swnote.java; /** * 复合操作例子 * * @author lzj * @date [2019-04-27] */ public class StatisticTest { private volatile int num = 0; public static void main(String[] args) { StatisticTest test = new StatisticTest(); test.statistic(); } public void statistic() { for (int i = 0; i < 20; i++) { new Thread(() -> { num++; }).start(); } System.out.println("num = " + num); } }
期望的运行结果是20,可是几乎每次运行结果都是不一样的,例如有的结果为:
这是因为num++这个操作不是原子性的,所以即使使用了volatile关键字,也是不能保证安全的。
关于如何在Java中使用volatile关键字就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。