博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ThreadLocal可能引起的内存泄露
阅读量:6654 次
发布时间:2019-06-25

本文共 899 字,大约阅读时间需要 2 分钟。

 threadlocal里面使用了一个存在弱引用的map,当释放掉threadlocal的强引用以后,map里面的value却没有被回收.而这块value永远不会被访问到了. 所以存在着内存泄露. 最好的做法是将调用threadlocal的remove方法.

  在threadlocal的生命周期中,都存在这些引用. 看下图: 实线代表强引用,虚线代表弱引用.

  

  每个thread中都存在一个map, map的类型是ThreadLocal.ThreadLocalMap. Map中的key为一个threadlocal实例. 这个Map的确使用了弱引用,不过弱引用只是针对key. 每个key都弱引用指向threadlocal. 当把threadlocal实例置为null以后,没有任何强引用指向threadlocal实例,所以threadlocal将会被gc回收. 但是,我们的value却不能回收,因为存在一条从current thread连接过来的强引用. 只有当前thread结束以后, current thread就不会存在栈中,强引用断开, Current Thread, Map, value将全部被GC回收.

  所以得出一个结论就是只要这个线程对象被gc回收,就不会出现内存泄露,但在threadLocal设为null和线程结束这段时间不会被回收的,就发生了我们认为的内存泄露。其实这是一个对概念理解的不一致,也没什么好争论的。最要命的是线程对象不被回收的情况,这就发生了真正意义上的内存泄露。比如使用线程池的时候,线程结束是不会销毁的,会再次使用的。就可能出现内存泄露。  

  PS.Java为了最小化减少内存泄露的可能性和影响,在ThreadLocal的get,set的时候都会清除线程Map里所有key为null的value。所以最怕的情况就是,threadLocal对象设null了,开始发生“内存泄露”,然后使用线程池,这个线程结束,线程放回线程池中不销毁,这个线程一直不被使用,或者分配使用了又不再调用get,set方法,那么这个期间就会发生真正的内存泄露。

转载地址:http://yqtto.baihongyu.com/

你可能感兴趣的文章
Spring 容器中的Bean 的生命周期
查看>>
关于swiper轮播图
查看>>
造成HashMap非线程安全的原因
查看>>
【本人秃顶程序员】在Java中使用函数范式提高代码质量
查看>>
机器学习和人工智能有什么区别?未来人们的看法可能和我们大相径庭
查看>>
IT兄弟连 JavaWeb教程 经典案例3
查看>>
MySQL 读写分离
查看>>
jsp里获取相对与绝对路径
查看>>
PHP开发入门1
查看>>
OSChina 周五乱弹 ——程序员会喜欢的 12 款键盘
查看>>
OSChina 周日乱弹 —— 别嫁出去霍霍别人了
查看>>
Python学习2-列表和元组
查看>>
linux环境下安装jdk1.8
查看>>
mysql基础知识
查看>>
数据挖掘topic
查看>>
php xdebug 配置
查看>>
runtime的使用
查看>>
iOS开发 BOOL / bool / Boolean / NSCFBoolean
查看>>
js常用数值计算
查看>>
elasticsearch
查看>>