2012-03-25 52 views
3

我正在用tomcat和jax-ws構建ws。 當我停止我的應用程序,通常在從日食重新部署,我得到這個消息:tomcat 7.0和jax-ws 2.2.5內存泄漏

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 [[email protected]deaa13]) and a value of type [com.sun.xml.internal.stream.XMLInputFactoryImpl] (value [[email protected]]) 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$1jaxws-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

你可以在調試中運行你的應用服務器,連接到它,把斷點放到'ThreadLocal'類方法並進行部署嗎?你會發現原因。 – 2012-03-27 21:10:23

回答

2

您使用(在JAX-WS實現)庫創造了一個線程局部,其關鍵是,已經裝入您的Web應用程序的類加載器的類型。

這意味着Web應用程序類加載器不能被垃圾收集,因爲如果沒有強引用其類的實例,則只能垃圾回收類加載器。多次重新部署後,您的JVM將耗盡perm gen空間。

真正的修復方法是監聽Web應用程序的生命週期,並在關閉時刪除線程本地。然而,因爲你不知道的JAX-WS實現的內部,這將是你很難做到這一點。

另一種解決方案是將jaxws-rt.jar(也可能是與JAX-WS相關的其他JAR文件)添加到Tomcat安裝的「lib」目錄中。在這種情況下,他們將共同類加載器加載,當您重新部署Web應用程序不會重新加載。

+0

是能夠監聽JAX-WS的生命週期,而不打補丁庫? – 2012-03-27 08:09:06

+0

和另一件事:爲什麼線程類是由不同的類加載器加載?非常感謝你:) – 2012-03-27 08:17:38

+0

你將不得不聽Servlet的生命週期,但它可能不會是直着捅到JX-WS實現的內部,實現線程局部釋放。它不是由不同的類加載器加載的線程類,而是線程本地的類型類。 – 2012-03-27 08:57:49