四川铁科建设监理有限公司官方网站,天津网站开发工资水平,机场建设相关网站,我做外贸要开国际网站吗ThreadLocalThreadLocalMapgetsetremove内存泄漏key用强/弱引用entry继承了弱引用ThreadLocal 一个对象的所有线程会共享其全局变量——线程不安全 解决方式#xff1a; 方式一#xff1a;同步机制#xff0c;加锁#xff08;时间换空间#xff09; 方式二#xff1a…ThreadLocalThreadLocalMapgetsetremove内存泄漏key用强/弱引用entry继承了弱引用ThreadLocal 一个对象的所有线程会共享其全局变量——线程不安全 解决方式 方式一同步机制加锁时间换空间 方式二ThreadLocal在每个线程中都创建了一个变量的副本线程各自访问自己的副本变量则不存在变量共享问题。空间换时间 变量与线程的关系private static所以ThreadLocal与线程的关系private static egsession 同一个线程内多个函数/组件间公共变量传递复杂降低了多模块间的耦合度。关联线程和线程上下文
ThreadLocalMap
每一个thread都有一个threadLocalMap变量keythreadLocal value值 一个thread可以有多个threadLocal
public class Thread implements Runnable {//存储与此线程相关的threadLocal值由threadLocal维护ThreadLocal.ThreadLocalMap threadLocals;
}jdk1.7的实现方式为每一个threadLocal都有一个threadLocalMap变量keythreadID value值
储存的entry减少之前由thread数量决定现在由threadLocal决定thread销毁对应的threadLocalMap也销毁。threadLocalMap的生命周期线程的生命周期内存使用减少
get private T get(Thread t) {ThreadLocalMap map getMap(t);if (map ! null) {if (map ThreadLocalMap.NOT_SUPPORTED) {return initialValue();} else {//根据ThreadLocalHash计算index取得Entry ThreadLocalMap.Entry e map.getEntry(this);if (e ! null) {SuppressWarnings(unchecked)T result (T) e.value;return result;}}}return setInitialValue(t);
}private T setInitialValue(Thread t) {T value initialValue();ThreadLocalMap map getMap(t);assert map ! ThreadLocalMap.NOT_SUPPORTED;if (map ! null) {map.set(this, value);} else {createMap(t, value);}if (this instanceof TerminatingThreadLocal? ttl) {TerminatingThreadLocal.register(ttl);}return value;
}set private void set(Thread t, T value) {ThreadLocalMap map getMap(t);if (map ThreadLocalMap.NOT_SUPPORTED) {throw new UnsupportedOperationException();}if (map ! null) {map.set(this, value);} else {createMap(t, value);}
}//存入值
private void set(ThreadLocal? key, Object value) {// We dont use a fast path as with get() because it is at// least as common to use set() to create new entries as// it is to replace existing ones, in which case, a fast// path would fail more often than not.Entry[] tab table;int len tab.length;int i key.threadLocalHashCode (len-1);for (Entry e tab[i];e ! null;e tab[i nextIndex(i, len)]) {if (e.refersTo(key)) {e.value value;return;}if (e.refersTo(null)) {replaceStaleEntry(key, value, i);return;}}tab[i] new Entry(key, value);int sz size;if (!cleanSomeSlots(i, sz) sz threshold)rehash();
}//初始化map
void createMap(Thread t, T firstValue) {t.threadLocals new ThreadLocalMap(this, firstValue);
}remove
private void remove(Thread t) {ThreadLocalMap m getMap(t);if (m ! null m ! ThreadLocalMap.NOT_SUPPORTED) {m.remove(this);}
}内存泄漏
引用链threadRef-thread-threadLocalMapthreadLocal,value-entry-value
弱引用如果这个对象只存在弱引用那么在下一次GC时会被清理。
ThreadLocalMap 中使用的 key 为 ThreadLocal 的弱引用。所以如果 ThreadLocal 没有被外部强引用在GC时ThreadLocal 会被清理掉ThreadLocalMap中使用这个 ThreadLocal 的 key 也会被设为null。由于value 是强引用不会被清理依然存在就会出现 key 为 null 的 valuevalue无法访问。在map中value为强引用不会被清除直到map被清除。然而由于threadLocalMap的生命周期线程线程池通常采取线程复用的方法所以线程池中的线程很难结束。所以value可能一直无法回收。
key用强/弱引用
key用强引用引用ThreadLocal的对象被回收了但threadLocalMap中还有threadLocal的强引用若未手动删除threadLocal不会被回收导致entry内存泄漏key用弱引用引用ThreadLocal的对象被回收了但threadLocalMap中还有threadLocal的弱引用即使未手动删除threadLocal也会被回收导致value内存泄漏结论若未手动删除key都会导致内存泄漏。强引用threadLocal不会被回收弱引用value。 内存泄漏与强/弱引用无关根本原因在于threadLocalMap的生命周期线程。
解决方式用完threadLocal调用remove清除无用的entry由于Entry继承了弱引用类会在下次GC时被JVM回收。get()set()remove()会顺便移除keynull的entry
entry继承了弱引用
public class ThreadLocalT {static class ThreadLocalMap {private Entry[] table;//轻量级map桶中存entry而非entry链表//Entry 继承弱引用static class Entry extends WeakReferenceThreadLocal? {/** The value associated with this ThreadLocal. */Object value;Entry(ThreadLocal? k, Object v) {super(k);value v;}}}
}