面试题——ThreadLocal

1.ThreadLocal是什么?

  • 当创建一个 ThreadLocal 变量时,每个访问它的线程都会复制这个变量到自己的本地内存,当多个线程操作这个变量时,实际操作的是自己本地内存里的变量,从而避免了线程安全问题。
  • 它的具体原理是:线程内部有一个由 ThreadLocal 维护的 ThreadLocalMap ,它类似于HashMap,变量其实是放在了当前线程的 ThreadLocalMap 中,而 ThreadLocal 只是传递了变量值。
  • ThreadLocalMap 存储以 ThreadLocal 为 key 的键值对,这个 key 是弱引用,也是 ThreadLocal 对象本身,而 value 存的是 ThreadLocal 对象调用 set 方法所设置的值。由于 ThreadLocalMap 的 key 是弱引用,而 Value 是强引用,这就导致在发生GC时弱引用Key被回收,而Value不会被回收,此时如果创建ThreadLocal的线程一直持续运行,那么value就可能一直得不到回收,从而发生内存泄漏。所以我们使用完 ThreadLocal 方法后,要注意在重写的afterCompletion方法中调用remove()方法进行手动清除。

注意知识点在面试过程中的串讲:

  1. 如果问到 ThreadLocal,那么顺带解释什么是强弱引用,以及另外两种引用。强软弱虚
  2. 如果问到内存泄漏案例,那么顺带解释 ThreadLocal 的原理,以及强弱引用。
为什么 ThreadLocalMap 的 key 是弱引用?
  • 如果 key 是强引用,那么发生 GC 时 ThreadLocalMap 还持有 ThreadLocal 的强引用,会导致 ThreadLocal 不会被回收,从而导致内存泄漏。弱引用 ThreadLocal 不会内存泄漏,对于 value 可以通过调用 set、get、remove 方法来清除,这算是最优的解决方案。

2.ThreadLocal的应用场景

ThreadLocal适用于独立变量副本的情况,如果变量为全局共享的,则不适用了。一个典型的应用场景是数据库连接管理,每个线程访问数据库都应当是一个独立的Session会话,如果多个线程共享同一个Session会话,有可能出现其他线程关闭连接了,而当前线程再执行提交时出现会话已关闭的异常。如果采取ThreadLocal的方式,能避免线程争抢Session,提高并发下的安全性。

3. 为什么ThreadLocal变量要设置成 static 修饰?

如果把ThreadLocal声明为某个类的非静态变量,也就是实例变量,那么每次创建该类的实例时,都会导致创建等同的对象,这些重复的对象会导致浪费。

4、思考扩展:为什么不将value也设置为弱引用?

因为不清楚这个value除了map的引用外,是否还存在其它引用,如果将它设为弱引用被GC干掉后,就容易导致value为null的错误。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值