ThreadLocal数据结构
ThreadLocal是一个线程本地变量,数据结构是存放在Thread中的一个ThreadLocal.ThreadLocalMap threadlocals,这是一个简单的Hash结构Map,初始容量也是16,2倍扩容,负载因子为2/3。与HashMap不同在于发生hash冲突时不是采用链表解决,而是使用线性探测方式(往前+1)解决。ThreadLocal.ThreadLocalMap 的key为this,当前ThreadLocal对象,hashCode为静态自增AtomicInteger自增生成,减少hash冲突。
ThreadLocal内存泄漏问题
ThreadLocalMap的key是弱引用类型。如果弱引用类型在下一次gc的时候将会回收,如果此时作为key的ThreadLocal被回收了,TheadLocal获取值的时候找不到对应的key值,但此时value值是强引用类型未被回收,导致该value无法使用与回收,导致内存泄漏,为防止该问题,ThreadLocal在调用get()获取不到对象时清空所有key为null的Entry,防止内存泄漏。
为什么ThreadLocalMap的key是弱引用类型
如果ThreadLocalMap的key为强引用类型,当ThreadLocal置空被回收时由于ThradLocalMap还引用着ThreadLocal,导致gc无法回收,导致内存泄漏,为防止该问题,ThreadLocalMap的key采用弱引用类型,如果ThreadLocal置空被回收时,由于TheadLocalMap的key是弱引用类型,下一次gc回收时会清空key,ThreadLoca没有了引用l也会被回收。
ThreadLocal使用场景
数据库连接,hibernate session,spring transational ,spring security session都基于ThreadLocal实现线程间隔离;