java volatile进阶(一)

本篇文章继续学习volatile。上篇文章简单的介绍了volatile和synchonized,这篇文章讲一下什么时候可以用volatile。

先看一段代码。

复制代码
package com.chzhao.voltiletest;

public class ChangeValue extends Thread {

    public void run() {
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        GlobalValues.STATUS = "stop";
    }
}
复制代码
复制代码
package com.chzhao.voltiletest;

public class VolatileDeep extends Thread {

    public void run() {
        while (true) {
            try {
                Thread.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            if (GlobalValues.STATUS.equals("stop")) {
                System.out.println(this.getName() + "stop");
                GlobalValues.STATUS = "running";
            }
        }
    }

    public static void main(String[] args) {
        Thread t1 = new VolatileDeep();
        t1.setName("t1");
        t1.start();

        Thread t2 = new VolatileDeep();
        t2.setName("t2");
        t2.start();

        Thread t3 = new ChangeValue();
        t3.start();
    }
}
复制代码
package com.chzhao.voltiletest;

public class GlobalValues {
    public static String STATUS = "running";
}

这段代码很简单,输出的是什么呢?

输出的是 t1stop?

还是t2stop?

还是

t1stop
t2stop

?

真实的情况是以上三种都有可能。

因为变量STATUS不是线程安全的,做为一个状态标识,在多线程的情况下状态是不可知的。

但是,如果在变量前面加上volatile关键字。

package com.chzhao.voltiletest;

public class GlobalValues {
    public volatile static String STATUS = "running";
}

每次的都会输出

t1stop
t2stop

当然,顺序不定。

 

这说明,如果加了volatile关键字,作为标识状态是能够保证线程安全的。为什么呢?

《Java 理论与实践: 正确使用 Volatile 变量》中介绍:

Volatile 变量具有 synchronized 的可见性特性,但是不具备原子特性。这就是说线程能够自动发现 volatile 变量的最新值。Volatile 变量可用于提供线程安全,但是只能应用于非常有限的一组用例:多个变量之间或者某个变量的当前值与修改后值之间没有约束。

《深入理解Java虚拟机:JVM高级特性与最佳实践》中介绍:

当一个变量定义为volatile之后,它将具备两种特性,第一是保证此变量对所有线程的可见性,这里的“可见性”是指当一条线程修改了这个变量的值,新值对于其他线程来说是可以立即得知的。而普通变量不能做到这一点,普通变量的值在线程间传递均需要通过主内存来完成,例如,线程A修改一个普通变量的值,然后向主内存进行回写,另外一条线程B在线程A回写完成了之后再从主内存进行读取操作,新变量值才会对线程B可见。

关于主内存,以后会有文章介绍。

由上面两段文字可以知道,volatile有一个很好的特性-可见性,可以理解可见性是所有线程均可以立即得到变量的值。所以,在多线程情况下,用volatile修饰状态值是非常适合的。

 

参考资料:

《Java 理论与实践: 正确使用 Volatile 变量》

《深入理解Java虚拟机:JVM高级特性与最佳实践》

posted @   wardensky  阅读(252)  评论(0编辑  收藏  举报
努力加载评论中...
点击右上角即可分享
微信分享提示