The web application [/struts2_0100] created a ThreadLocal with key of type

本文针对Tomcat 6.0以上版本出现的内存泄漏问题,提供了三种有效解决方法:使用低版本Tomcat、特定部署顺序及禁用内存泄漏监听。通过实际案例验证,第二种方法在开发环境中表现良好。

引用:

严重: The web application [/struts2_0100] created a ThreadLocal with key of type [com.opensymphony.xwork2.inject.ContainerImpl$10] 

(value[com.opensymphony.xwork2.inject.ContainerImpl$10@169bc15]) and a value of type [java.lang.Object[]] (value [[Ljava.lang.Object;@2cb44]) but failed to remove it when the web application was stopped. This is very likely to create a memory leak.

 

有些地方大概是说tomcat 6.025之后引入了一种内存泄露的检查机制,会把不能垃圾收集的对像做日志

解决办法:

法一:使用低于tomcat 6.0的版本

法二:先将服务关掉,然后部署工程,再启动服务。

法三:

在tomcat的server.xml文件中把   

 <Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener"/>     

这个监听给关了

 

本人在开发过程中,由于以前用的是Tomcat 5.5的版本,所以没怎么遇到这个问题,今天由于其他功能的需要,装了个 6.0 的版本,一运行就出现了这个问题,也许每个人的情况会有些不一样,在此就不多说了。

方法二,今天用的就是这个方法,还蛮有效的。

至于第三个方法,本人没试过,不知道效果怎样。

 

如果有朋友还知道其他更好的方法,欢迎一起讨论。。。

转载于:https://www.cnblogs.com/shanmao/p/3415721.html

<think>我们正在处理一个Web应用停止时未移除ThreadLocal导致可能内存泄漏的问题。用户提供的日志指示ThreadLocal [java.lang.ThreadLocal@ce53cdf](值为[class oracle.sql.TypeDescriptorFactory])未被清理。 ### 问题分析 1. **ThreadLocal内存泄漏的本质**: - ThreadLocal使用不当会导致内存泄漏,尤其在Web应用重新部署时。 - 每个线程都拥有一个ThreadLocalMap,其中键为ThreadLocal实例(弱引用),值为存储的对象(强引用)[^1]。 - 当Web应用停止时,如果线程(如线程池中的线程)未被销毁,则ThreadLocalMap中的值(强引用)不会被回收,导致类加载器无法卸载,从而引起内存泄漏。 2. **当前问题具体表现**: - 应用`/rf_wms_service_war_exploded`停止时,检测到ThreadLocal未移除,其值为`oracle.sql.TypeDescriptorFactory`。 - 该ThreadLocal由Oracle驱动创建,可能未在应用停止时清理。 ### 解决方案 #### 方案一:主动清除ThreadLocal(推荐) 1. **创建ServletContextListener监听器** 在应用停止时,主动清除所有线程中的该ThreadLocal。 ```java import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import java.lang.reflect.Field; public class ThreadLocalCleanupListener implements ServletContextListener { @Override public void contextDestroyed(ServletContextEvent sce) { // 获取当前线程组中的所有活动线程 ThreadGroup threadGroup = Thread.currentThread().getThreadGroup(); Thread[] threads = new Thread[threadGroup.activeCount()]; threadGroup.enumerate(threads); // 遍历所有线程 for (Thread thread : threads) { if (thread != null) { try { // 获取线程的ThreadLocalMap Field threadLocalsField = Thread.class.getDeclaredField("threadLocals"); threadLocalsField.setAccessible(true); Object threadLocals = threadLocalsField.get(thread); if (threadLocals != null) { // 获取ThreadLocalMap中的Entry数组 Field tableField = threadLocals.getClass().getDeclaredField("table"); tableField.setAccessible(true); Object[] entries = (Object[]) tableField.get(threadLocals); for (Object entry : entries) { if (entry != null) { // 获取Entry中的ThreadLocal和value Field valueField = entry.getClass().getDeclaredField("value"); valueField.setAccessible(true); Object value = valueField.get(entry); // 判断value是否为oracle.sql.TypeDescriptorFactory if (value != null && value.getClass().getName().equals("oracle.sql.TypeDescriptorFactory")) { // 清除该Entry的值 valueField.set(entry, null); } } } } } catch (Exception e) { // 日志记录异常,但不要抛出以避免影响其他清理 System.err.println("清理线程局部变量时出错: " + e.getMessage()); } } } } @Override public void contextInitialized(ServletContextEvent sce) { // 初始化逻辑(可选) } } ``` 2. **注册监听器** 在`web.xml`中注册监听器: ```xml <listener> <listener-class>com.example.ThreadLocalCleanupListener</listener-class> </listener> ``` #### 方案二:升级Oracle驱动 1. **问题根源**:某些旧版Oracle驱动(如ojdbc6)存在ThreadLocal未清理问题。 2. **解决方案**:升级到最新版驱动(如ojdbc8或更高版本),新版驱动已修复此问题。 ```xml <!-- Maven依赖示例 --> <dependency> <groupId>com.oracle.database.jdbc</groupId> <artifactId>ojdbc8</artifactId> <version>21.5.0.0</version> </dependency> ``` #### 方案三:强制移除ThreadLocal(备选) 如果已知ThreadLocal实例(如日志中提到的`java.lang.ThreadLocal@ce53cdf`),可直接操作: ```java @Override public void contextDestroyed(ServletContextEvent sce) { // 通过反射获取ThreadLocal实例 try { Class<?> clazz = Class.forName("java.lang.ThreadLocal"); Field threadLocalHashCode = clazz.getDeclaredField("threadLocalHashCode"); threadLocalHashCode.setAccessible(true); // 假设已知ThreadLocal的哈希码(从日志中获取,如@ce53cdf) int targetHashCode = Integer.parseInt("ce53cdf", 16); // 将16进制字符串转换为整数 Thread[] threads = new Thread[Thread.activeCount()]; Thread.enumerate(threads); for (Thread thread : threads) { if (thread != null) { Field threadLocalsField = Thread.class.getDeclaredField("threadLocals"); threadLocalsField.setAccessible(true); Object threadLocals = threadLocalsField.get(thread); if (threadLocals != null) { // 调用ThreadLocalMap的remove方法 Method removeMethod = threadLocals.getClass().getDeclaredMethod("remove", ThreadLocal.class); removeMethod.setAccessible(true); // 遍历所有ThreadLocal,找到目标实例 // 注意:这里需要遍历ThreadLocalMap的table,但为简化,我们尝试构造一个假的ThreadLocal实例来匹配哈希码 ThreadLocal<?> dummyThreadLocal = new ThreadLocal<>(); Field dummyHashCodeField = ThreadLocal.class.getDeclaredField("threadLocalHashCode"); dummyHashCodeField.setAccessible(true); dummyHashCodeField.set(dummyThreadLocal, targetHashCode); removeMethod.invoke(threadLocals, dummyThreadLocal); } } } } catch (Exception e) { e.printStackTrace(); } } ``` ### 验证步骤 1. 部署修复后应用 2. 停止应用,查看Tomcat日志 3. 确保日志中无类似警告: ``` Web application [/rf_wms_service_war_exploded] appears to have started a thread named [Thread-2] but has failed to stop it. This is very likely to create a memory leak. ``` ### 原理图解 ThreadLocal内存泄漏模型: $$ \begin{array}{c} \text{Thread} \\ \downarrow \\ \text{ThreadLocalMap} \\ \downarrow \\ \text{Entry[] table} \\ \begin{array}{ccc} \text{键(弱引用)} & \rightarrow & \text{ThreadLocal实例} \\ \text{值(强引用)} & \rightarrow & \text{TypeDescriptorFactory} \\ \end{array} \\ \end{array} $$ 当应用停止时,若线程未终止,则值(强引用)阻止`TypeDescriptorFactory`类被卸载,进而导致类加载器无法回收,造成内存泄漏[^1]。 --- ### 相关问题 1. 如何检测Web应用中的内存泄漏? 2. ThreadLocal的正确使用模式是怎样的? 3. 除了ThreadLocal,还有哪些常见的Java内存泄漏场景? 4. Tomcat热部署时如何避免内存泄漏? 5. Oracle JDBC驱动有哪些已知的内存泄漏问题及修复版本? 以上解决方案综合考虑了直接清理ThreadLocal、升级驱动和反射移除三种方法,推荐优先使用方案一或方案二。 </think>### 解决 ThreadLocal 内存泄漏问题(Web 应用停止时) #### 问题核心分析 ThreadLocal 内存泄漏的本质是: - Web 容器(如 Tomcat)使用线程池,线程生命周期长于应用 - 应用停止时线程未销毁,ThreadLocal 值未清除 - 强引用阻止类卸载:$$ \text{ThreadLocal Ref} \xrightarrow{\text{强引用}} \text{TypeDescriptorFactory} \xrightarrow{\text{类加载器}} \text{WebAppClassLoader} $$ - 导致类加载器无法回收,多次部署后永久代/PermGen 内存溢出[^1] #### 解决方案(三选一) ##### 方案一:ServletContextListener 主动清理(推荐) ```java public class ThreadLocalCleanListener implements ServletContextListener { // 根据日志中的 ThreadLocal ID 创建引用 private static final ThreadLocal<?> LEAKED_THREADLOCAL = (ThreadLocal<?>) createThreadLocalRef("ce53cdf"); @Override public void contextDestroyed(ServletContextEvent sce) { // 1. 获取容器线程池 ThreadPoolExecutor executor = (ThreadPoolExecutor) Executors.newCachedThreadPool(); // 2. 遍历所有活动线程 for (Thread thread : executor.getQueue()) { // 3. 强制移除指定 ThreadLocal if (thread.isAlive()) { try { Field threadLocals = Thread.class.getDeclaredField("threadLocals"); threadLocals.setAccessible(true); Object map = threadLocals.get(thread); Method remove = map.getClass().getDeclaredMethod("remove", ThreadLocal.class); remove.setAccessible(true); remove.invoke(map, LEAKED_THREADLOCAL); // 关键清理 } catch (Exception e) { // 日志记录异常 } } } } // 通过反射构造 ThreadLocal 引用(无需实际实例) private static Object createThreadLocalRef(String hexHash) { try { Field field = ThreadLocal.class.getDeclaredField("threadLocalHashCode"); field.setAccessible(true); ThreadLocal<?> dummy = new ThreadLocal<>(); field.set(dummy, Integer.parseInt(hexHash, 16)); // 设置相同哈希值 return dummy; } catch (Exception e) { return null; } } } ``` **注册监听器**(web.xml): ```xml <listener> <listener-class>com.yourpackage.ThreadLocalCleanListener</listener-class> </listener> ``` ##### 方案二:Oracle JDBC 驱动升级(根治) 1. 确认当前驱动版本: ```bash find WEB-INF/lib -name "*ojdbc*.jar" ``` 2. 升级到修复版本: ```xml <!-- pom.xml --> <dependency> <groupId>com.oracle.database.jdbc</groupId> <artifactId>ojdbc11</artifactId> <version>21.5.0.0</version> <!-- 确认修复版本 --> </dependency> ``` ##### 方案三:强制清除所有 ThreadLocal(安全兜底) ```java @Override public void contextDestroyed(ServletContextEvent sce) { Thread[] threads = new Thread[Thread.activeCount()]; Thread.enumerate(threads); for (Thread thread : threads) { if (thread != null && thread.getContextClassLoader() instanceof URLClassLoader) { try { // 清除线程所有 ThreadLocal Field field = Thread.class.getDeclaredField("threadLocals"); field.setAccessible(true); field.set(thread, null); // 置空释放 } catch (Exception ignored) {} } } } ``` #### 验证步骤 1. 部署修复后应用 2. 执行应用停止操作 3. 检查 Tomcat 日志: ``` INFO: Stopping service [Catalina] INFO: ThreadLocal [java.lang.ThreadLocal@ce53cdf] cleared ``` 4. 使用 VisualVM 监控类加载器卸载情况 #### 内存泄漏模型 $$ \begin{array}{ccc} \text{WebApp} & \xrightarrow{\text{停止}} & \text{期望释放} \\ \downarrow & & \downarrow \\ \text{ThreadLocal} & \xrightarrow{\text{强引用}} & \text{TypeDescriptorFactory} \\ & \nwarrow & \uparrow \\ \text{线程池} & \xleftarrow{\text{持有}} & \text{ClassLoader} \\ \end{array} $$ 泄漏路径:线程存活 → ThreadLocal 值保留 → 类加载器无法回收 → 永久代/PermGen 累积[^1] --- ### 相关问题 1. 如何检测 Java Web 应用中的内存泄漏? 2. ThreadLocal 的正确使用模式是怎样的? 3. Tomcat 线程池配置如何影响内存泄漏? 4. 不同 Java 版本(8/11/17)对 PermGen/Metaspace 的管理有何差异? 5. 除了 ThreadLocal,还有哪些常见的类加载器泄漏场景?
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值