ThreadLocal原理和 java类全局静态变量在多线程中数据混乱问题


引自:ThreadLocal原理和 java类全局静态变量在多线程中数据混乱问题 - X凯 - 博客园

一般想要在不同方法中传递上下文数据,会使用全局变量,而想在不同类中传递上下文数据,则会使用全局静态变量,因为静态全局变量是属于类一直存在的,但是在多线程操作状态下,java类全局静态变量在多线程中会出现数据混乱问题,因为多个线程同时对同一个静态变量操作时,无法确保每个线程取出的值是自己放的值。
这时就出现了ThreadLocal
ThreadLocal众所周知,可以把变量绑定到到某一线程上,这个thread里有个静态内部类(可以理解为一个全局变量ThreadLocalMap),

 

 其泛型为<ThreadLocal,Object> 

而threadlocal在执行set方法时会先获取当前线程(Thread t = Thread.currentThread()),使用当前线程去拿到一个ThreadLocalMap,如果这个map不为空,说明当前线程之前有绑定过的map,就更新这个这个map的value,如果没有就以自身为key把值放到map中,从而实现变量与线程的绑定。

 

 关于ThreadLocal的用法,有这两种说法

1.传递上下文变量,减少程序复杂度

2.空间换时间,解决并发下对临界资源的访问。

其次还要注意的是:ThreadLocal和同步机制synchonzied相比

(Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。)

1.1.synchonzied同步机制是为了实现同步多线程对相同资源的并发访问控制。同步的主要目的是保证多线程间的数据共享。同步会带来巨大的性能开销,所以同步操作应该是细粒度的(对象中的不同元素使用不同的锁,而不是整个对象一个锁)。如果同步使用得当,带来的性能开销是微不足道的。使用同步真正的风险是复杂性和可能破坏资源安全,而不是性能。 

2.ThreadLocal以空间换取时间,提供了一种非常简便的多线程实现方式。因为多个线程并发访问无需进行等待,所以使用ThreadLocal会获得更大的性能。

3.ThreadLocal中的对象,通常都是比较小的对象。另外使用ThreadLocal不能使用原子类型,只能使用Object类型。ThreadLocal的使用比synchronized要简单得多。 

4.synchronized是利用锁的机制,使变量或代码块在某一时该只能被一个线程访问。而ThreadLocal为每一个线程都提供了变量的副本,使得每个线程在某一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的数据共享。而Synchronized却正好相反,它用于在多个线程间通信时能够获得数据共享。 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java多线程编程,如果多个线程同时访问和修改同一个全局变量,可能会导致读取值混乱问题,这是由于线程之间的竞争条件引起的。具体来说,当一个线程正在读取全局变量的值时,另一个线程可能会修改该变量的值,导致读取到的值不一致或者不正确。 为了解决这个问题,可以采取以下几种方式: 1. 使用synchronized关键字:可以使用synchronized关键字来保证在同一时间只有一个线程可以访问和修改全局变量。通过在读取和修改全局变量的方法或代码块上添加synchronized关键字,可以确保线程安全。 2. 使用volatile关键字:可以使用volatile关键字来修饰全局变量,它可以确保多个线程之间对该变量的可见性。当一个线程修改了volatile变量的值时,其他线程可以立即看到最新的值。 3. 使用Lock对象:可以使用Lock对象来实现对全局变量的访问控制。通过在读取和修改全局变量的代码块使用Lock对象进行加锁和解锁操作,可以确保同一时间只有一个线程可以访问和修改全局变量。 4. 使用ThreadLocalThreadLocal可以实现每个线程都拥有自己独立的全局变量副本。每个线程对全局变量的修改只会影响到自己的副本,不会影响其他线程的副本,从而避免了读取值混乱问题。 总之,在多线程编程,为了避免全局变量读取值混乱问题,需要采取适当的同步机制或者使用线程本地变量来保证线程安全。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值