ThreadLocal和volatile关键字的区别

2 篇文章 0 订阅
1 篇文章 0 订阅

最近在学习了ThreadLocal后,了解到它是在线程间做数据隔离的。这个时候我就想起了volatile保证变量的可见,一时间就有点混乱了,想着为什么数据又要隔离又要保证可见性,这是什么个鬼。后面重新梳理了下,我才正在理解两个东西的用法和使用场景。

首先ThreadLocal是为了保障数据隔离,避免不同线程对该变量被通过对外提供的方法进行修改,导致了某个线程 set 以后,想要 get ,发现和之前 set的值不一致。导致业务处理异常

而volatile第一要务是保证变量的可见性,也就是说,所有线程无论何时都是获取到变量的最新值(如果不加不能保证一定获取变量的最新值)

关键点:
ThreadLocal想让线程间对某个变量不相互干扰,隔离开来;
volatile是为了保证线程间拿到的变量都是同一个内容
我一开始没弄清在于多线程情况的变量的共享情况不熟悉,不加volatile的变量,它们也是会共享的,只是线程间不能实时同步,用volatile修饰后使的能够实时同步;我因为一直看volatile的可见性可见性,导致我以为只有被volatile修饰的变量才是线程共享,如果不加则是隔离,导致我在想为什么都是隔离的了还要使用ThreadLocal。

总结:
1.不被volatile修饰的变量在多线程环境下也是共享的
2.为了避免变量共享,通过threadlocal实现了数据隔离的作用
3.volatile保证的可见性是指共享变量实时同步,所有线程获取到的都是最新值

最后,我在测试volatile关键字的时候,发现在while循环中使用System.out.println()会让本地变量读取到主内存的变量?不明白原理

public class VolatileTest {
    private   int q = 0;

    public int getQ() {
        return q;
    }

    public void setQ(int q) {
        this.q = q;
    }

    public static void main(String[] args) throws InterruptedException {
        VolatileTest volatileTest = new VolatileTest();
        new Thread(()->{
            while (volatileTest.getQ()==0){
            //有了这一个就会跳出循环了。。去掉又可以死循环了
                System.out.println();
            }
        }).start();
        Thread.sleep(1000);
        volatileTest.setQ(1);
        System.out.println(volatileTest.getQ());
    }
}
  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值