深入jvm类卸载机制:强引用链与类加载器回收
本文深入探讨JVM的类卸载机制,重点分析类加载器与类之间的相互引用关系,并解答MyCounter.class和WebAppClassLoader是否会因循环引用导致内存泄漏的问题。
文中举例说明了强引用链:线程 -> ThreadLocalMap -> counter -> MyCounter.class -> WebAppClassLoader,导致WebAppClassLoader无法回收。 这引发了MyCounter.class卸载条件的疑问。 文章指出,类卸载的必要条件之一是加载该类的类加载器被回收。 这引出了核心问题:如果MyCounter.class由WebAppClassLoader加载,MyCounter.class是否会反过来强引用WebAppClassLoader,从而阻止其回收,最终导致MyCounter.class也无法卸载?
关键在于理解“无任何外部引用”的含义。 认为WebAppClassLoader回收的唯一条件是“没有任何引用”是不准确的。 即使存在MyCounter.class和WebAppClassLoader之间的循环引用,如果它们没有被其他对象(例如,程序中的其他类、线程等)直接或间接引用,JVM的垃圾回收器(例如标记清除算法)仍然可以识别并回收它们。 因为从GC Root(例如线程栈)出发,无法访问这两个对象。 因此,即使存在循环引用,只要没有其他强引用指向它们,它们仍然会被GC回收。
所以,MyCounter.class并不一定总是强引用WebAppClassLoader,阻止其回收。 知乎文章中提到的内存泄漏,更可能是由于线程未正确结束,导致ThreadLocal中持有的对象无法回收,最终导致WebAppClassLoader及其加载的类无法卸载。 问题的根源不在于类与类加载器的相互引用,而在于程序没有正确管理线程和资源,造成了内存泄漏。