volatile线程可见性

volatile线程可见性

什么是线程可见,一段代码带你了解线程可见性

public class VolatileDemo {

    int x = 0;
    /**
     * 这里的flag没有被volatile修饰
     */
    boolean flag = false;

    /**
     * 写操作
     */
    private void write() {
        x = 5;
        flag = true;
        System.out.println("x=>" + x);
        System.out.println("b =>" + flag);
    }

    /**
     * 读操作
     */
    private void read() {
        //如果flag=false的话,就会无限循环,直到flag=true才会执行结束,会打印出x的值
        while (!flag) {

        }
        System.out.println("x=" + x);
    }

    public static void main(String[] args) throws Exception {
        final VolatileDemo volatileDemo = new VolatileDemo();

        //线程1执行写操作
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                volatileDemo.write();
            }
        });

        //线程2执行读操作
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                volatileDemo.read();
            }
        });

        //我们让线程2的读操作先执行
        thread2.start();

        //睡1毫秒,为了保证线程2比线程1先执行
        TimeUnit.MILLISECONDS.sleep(1);

        //再让线程1的写操作执行
        thread1.start();

        //等待线程1和线程2全部结束后,打印执行结束
        System.out.println("执行结束");
    }
}

在这里插入图片描述

注意我们的flag没有用volatile修饰,我们先启动了线程2的读操作,后启动了线程1的写操作,由于线程1和线程2会保存x和flag的副本到自己的工作内存中,线程2执行后,由于他副本flag=false,所以会进入到无限循环中,线程1执行后修改的也是自己副本中的b=true,然而线程2无法立即察觉到,所以执行上面代码后,不会打印“执行结束”,因为线程2一直在执行!

给flag加了volatile关键字修饰后,线程1对flag做了修改,然后会立即更新内存中的值,线程2通过嗅探发现自己的副本已经过期了,然后重新从内存中拿到flag=true的值,然后跳出while循环,执行结束!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值