我正在用tomcat和jax-ws构建一个ws。 当我停止我的应用程序时,通常在从eclipse重新部署时,我收到此消息:
25-mar-2012 16.21.16 com.sun.xml.ws.transport.http.servlet.WSServletDelegate destroy
INFO: WSSERVLET15: JAX-WS servlet destroyed
25-mar-2012 16.21.16 com.sun.xml.ws.transport.http.servlet.WSServletContextListener contextDestroyed
INFO: WSSERVLET13: JAX-WS context listener destroyed
25-mar-2012 16.21.16 org.apache.catalina.loader.WebappClassLoader checkThreadLocalMapForLeaks
GRAVE: The web application [/xccm] created a ThreadLocal with key of type [com.sun.xml.ws.api.streaming.XMLStreamReaderFactory$Default$1] (value [com.sun.xml.ws.api.streaming.XMLStreamReaderFactory$Default$1@7edeaa13]) and a value of type [com.sun.xml.internal.stream.XMLInputFactoryImpl] (value [com.sun.xml.internal.stream.XMLInputFactoryImpl@4c700677]) but failed to remove it when the web application was stopped. Threads are going to be renewed over time to try and avoid a probable memory leak.
使用ClasspathHelper挖掘我发现com.sun.xml.ws.api.streaming.XMLStreamReaderFactory$Default$1
引用了jaxws-rt.jar
但是我没有在我的代码中明确使用任何jax-ws类,除了:
import javax.jws.WebParam;
import javax.jws.WebService;
import javax.xml.bind.JAXBContext;
据我所知tomcat加载com.sun.xml.ws.transport.http.servlet.WSServlet
(以及日志片段中的其他类)。
任何想法如何解决这个问题?
THX
答案 0 :(得分:3)
您使用的库(JAX-WS实现)已创建一个本地线程,其键的类型已由您的Web应用程序类加载器加载。
这意味着Web应用程序类加载器不能被垃圾收集,因为如果没有强引用其类的实例,则只能对类加载器进行垃圾回收。在多次重新部署之后,您的JVM将耗尽perm gen空间。
真正的解决方法是监听Web应用程序的生命周期,并在关闭时删除本地线程。但是,由于您不了解JAX-WS实现的内部结构,因此您很难做到这一点。
另一种解决方案是将jaxws-rt.jar(以及与JAX-WS相关的其他JAR文件)添加到Tomcat安装的“lib”目录中。在这种情况下,它们将由公共类加载器加载,并且在重新部署Web应用程序时不会重新加载。