I'm trying to clean up Tomcat warnings about memory leaks due to creating my own threads. http://wiki.apache.org/tomcat/MemoryLeakProtection proposes that I call myThread.setContextClassLoader(null)
prior to starting the thread.
What is the implication of this call? Will the code in the run() method still be able to resolve classes from my application?
Yes it will. The
Thread.getContextClassLoader()
is a mechanism for generic frameworks load resources from further down the class loader tree.Take Tomcat's class loader hierarchy.
The servlet or JSP framework resides in the
Common
class loader. If one of these frameworks is to load a classpath resource fromWebapp1
, they could try:But since class loading mechanics only delegates calls up the class loader chain, this would fail. This means that all frameworks that needs to load a resource instead do:
And the servlet container makes sure this is the
Webapp1
class loader whenever the thread is executing in that context. So the thread's context classloader is effectively a way for frameworks to load classes from "the wrong direction".When you spawn a new thread, that thread gets, by default the context class loader of its parent (your
Webapp1
classloader). If you consequently stopWebapp1
, tomcat is supposed to be able to GC that webapp, but is unable to do so as long as there's any reference left to theWebapp1
class loader - hence the warning.Good article about context class loaders.