前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >ThreadLocalMap里Entry为何声明为WeakReference

ThreadLocalMap里Entry为何声明为WeakReference

作者头像
JavaEdge
发布2018-05-16 15:48:30
1.6K0
发布2018-05-16 15:48:30
举报
文章被收录于专栏:JavaEdge

Java里,每个线程都有自己的ThreadLocalMap,里边存着自己私有的对象。Map的Entry里,key为ThreadLocal对象,value即为私有对象T。在spring MVC中,常用ThreadLocal保存当前登陆用户信息,这样线程在任意地方都可以取到用户信息了

跳过ThreadLocal和ThreadLocalMap的工作原理及场景不讲(看本人此专栏下另一篇文章),主要来说说Entry为什么是WeakReference

代码语言:javascript
复制
       /**
         * 此哈希表中的项继承了WeakReference
         * 它的主要引用域是key(这总是一个 ThreadLocal对象)。
         * 请注意,空key(即entry.get()== null) 表示该key不再被引用,所以该项可以从表中删除。
         * 这些被引用项的将作为下面代码中的“陈旧项”
         */
        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

先来看看WeakReference的作用,百度一下:

WeakReference是Java语言规范中为了区别直接的对象引用(程序中通过构造函数声明出来的对象引用)而定义的另外一种引用关系。 WeakReference标志性的特点是:reference实例不会影响到被应用对象的GC回收行为(即只要对象被除WeakReference对象之外所有的对象解除引用后,该对象便可以被GC回收),只不过在被对象回收之后,reference实例想获得被应用的对象时程序会返回null

我的理解就是,WeakReference对应用的对象users是弱引用,不会影响到users的GC行为。 如果是强引用的话,在线程运行过程中,我们不再使用users了,将users置为null,但users在线程的ThreadLocalMap里还有引用,导致其无法被GC回收(当然,可以等到线程运行结束后,整个Map都会被回收,但很多线程要运行很久,如果等到线程结束,便会一直占着内存空间)。 而Entry声明为WeakReference,users置为null后,线程的threadLocalMap就不算强引用了,users就可以被GC回收了。map的后续操作中,也会逐渐把对应的"stale entry"清理出去,避免内存泄漏

所以,我们在使用完ThreadLocal变量时,尽量用threadLocal.remove()来清除,避免threadLocal=null的操作。 前者remove()会同时清除掉线程threadLocalMap里的entry,算是彻底清除 而后者虽然释放掉了threadLocal,但线种threadLocalMap里还有其"stale entry",后续还需要处理

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018.02.11 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档