一. 释意及作用
ThreadLocal字面意思就是本地线程,那它的作用就是在多线程环境下不同线程之间实现数据的隔离.
二. 工作原理
每个线程都有一个ThreadLocalMap(容器),并且ThreadLocal中包含Entry对象,其对象的key是弱引用,那为什么它的key是弱引用呢?那就不得不讲讲java对象了
java对象:
强引用:触发GC回收的时候,java栈中使用的对象是无法被回收的,即使发生OOM也不会被清理(在下面会简单解释一下OOM).
软引用:触发GC回收的时候,内存告急将会把软引用清理.
弱引用:触发GC回收的时候,无论内存是否充足,都会将弱引用全部回收.
虚引用:触发GC回收的时候,会有通知机制(一般情况下不会使用).
结论:采用反证手法,如果采用的是强引用,ThreadLocal对象无法被回收,并且ThreadLocal是存在内存泄漏的问题,所有最终可能会导致OOM,但是采用弱引用,触发GC回收的时候,ThreadLocal对象就会被回收,至少就能保证ThreadLocal不会导致内存泄漏,但是即使将ThreadLocal设置为弱引用也无法从根本上解决内存泄漏问题,当Entry中key被清理后,并为null,但是对于的value却无法清理,并且也无法被访问到,更不能被GC回收,长时间的积累依然可能导致内存泄漏.
三. 那ThreadLocal如何使用才能避免内存的泄漏?
ThreadLocal调用set,get方法会默认将key为null的数据进行清理,当线程执行完毕后,将ThreadLocal中的数据手动清理也可以有效的避免内存泄漏.
四. OOM
OOM通常是指"Out Of Memory",字面意思就可理解,意味着内存耗尽,这可能是由于内存泄漏或者程序需要的内存超出了系统可用的内存限制。
-
内存泄漏(Memory Leak): 这是指程序中分配的内存没有被正确释放,导致程序持续占用更多的内存而不释放。随着时间的推移,累积的内存泄漏可能导致系统耗尽可用内存,最终导致 OOM 错误。
-
内存溢出(Out of Memory): 这是指程序在某一时刻请求的内存超过了系统当前可用的内存。这可能是因为程序要求的内存量很大,而系统无法提供足够的内存。
在实际情况中,OOM 错误可能是内存泄漏的结果,也可能是由于程序在某个时刻需要大量内存而导致的。要诊断和解决这类问题,通常需要使用工具来分析内存使用情况,查找潜在的内存泄漏,并优化代码以减少内存消耗。