volatile是否线程安全的

证明程序:

public class TestVolatile {

    public static volatile int numb = 0;

    public static void main(String[] args) throws Exception {
        for (int i = 0; i < 100; i++) {

            new Thread(i+""){

                @Override
                public void run() {
                    for (int i = 0; i < 1000; i++) {
                        numb=numb+1;
                    }
                }
            }.start();

        }
        
        Thread.sleep(2000);
        System.out.println(numb);
    }

}

 

正确结果:100000

事实是:

 

 

 

90000多,不固定

 

事实胜于雄辩,所以volatile不是线程安全的

 

原因:

jvm有一个主内存,volatile修饰的变量还是在主内存里面,而线程去对变量进行+运算,是在线程的工作内存修改的,不能立刻及时同步到主内存,多线程情况下,还是会存在看到变量不一样的情况 ,volatile修饰的变量,在有多线程对变量做运算并不安全。

专业解释:

修改volatile变量分为四步:

1)读取volatile变量到local

2)修改变量值

3)local值写回

4)插入内存屏障,即lock指令,让其他线程可见

这样就很容易看出来,前三步都是不安全的,取值和写回之间,不能保证没有其他线程修改。原子性需要锁来保证。

 

posted on 2020-07-07 21:48  又在桃花换酒钱  阅读(926)  评论(0编辑  收藏  举报