场景
- 在部署
jfinal
程序到tomcat
的时候,使用startup.bat
启动网站,之后用shutdown.bat
关闭网站,发现命令行窗口无法终止退出。报以下错误, 怎么处理?
06-Mar-2023 11:09:24.534 严重 [localhost-startStop-2] org.apache.catalina.loader
.WebappClassLoaderBase.checkThreadLocalMapForLeaks web应用程序[ROOT]创建了一个Th
readLocal,其键类型为[com.jfinal.template.io.WriterBuffer$1](值为[com.jfinal.te
mplate.io.WriterBuffer$1@36d6d87d]),值类型为[com.jfinal.template.io.ByteWriter
](值为[com.jfinal.template.io.ByteWriter@24e46eae),但在停止web应用程序时未能
将其删除。线程将随着时间的推移而更新,以尝试避免可能的内存泄漏
09-Mar-2023 08:49:59.960 WARNING [localhost-startStop-2] org.apache.catalina.loader.WebappClassLoaderBase.clearReferencesThreads The web application [ROOT] appears to have started a thread named [CaptchaCache] but has failed to stop it. This is very likely to create a memory leak. Stack traceof thread:
说明
问题1.
- 第一个错误是
jfinal
里使用了WriterBuffer
类,这个类用于缓存输出对象。 看它的源码,它创建了一个ThreadLocal
线程本地变量,这个变量的特点有:-
变量
byteWriters
只有它所属的线程可以访问到它的值,也就是ByteWriter
, 目的是为了在线程范围内重用ByteWriter
缓存。 -
注意: 有多少个线程调用
getByteWriter()
就有多少个ByteWriter
实例存储在每个线程里。即使tomcat
关闭,线程里仍然有这些实例的引用,因此这些实例就不会被释放。 -
解决办法就是不使用这个线程本地变量,实现一个
WriteBuffer
的子类。 -
注意: 这个线程本地变量重用缓存部分一般在服务器端控制大文件输出时才会用到, 输出
html
并不会重用,因此这个线程本地变量可以去掉。
-
WriterBuffer.java
package com.jfinal.template.io;
/**
* WriterBuffer
*/
public class WriterBuffer {
private static final int MIN_BUFFER_SIZE = 64; // 缓冲区最小 64 字节
private static final int MAX_BUFFER_SIZE = 1024 * 1024 * 2; // 缓冲区最大 2M 字节
private int bufferSize = 1024; // 缓冲区大小
private int reentrantBufferSize = 128; // 可重入缓冲区大小
private EncoderFactory encoderFactory = new EncoderFactory();
private final ThreadLocal<ByteWriter> byteWriters = new ThreadLocal<ByteWriter>() {
protected ByteWriter initialValue() {
return new ByteWriter(encoderFactory.getEncoder(), bufferSize);
}
};
...
public ByteWriter getByteWriter(java.io.OutputStream outputStream) {
ByteWriter ret = byteWriters.get();
if (ret.isInUse()) {
ret = new ByteWriter(encoderFactory.getEncoder(), reentrantBufferSize);
}
return ret.init(outputStream);
}
...
}
MyWriterBuffer.java
import com.jfinal.template.io.*;
public class MyWriterBuffer extends WriterBuffer {
private int bufferSize = 1024; // 缓冲区大小
private int reentrantBufferSize = 128; // 可重入缓冲区大小
protected EncoderFactory encoderFactory = new EncoderFactory();
protected final ThreadLocal<ByteWriter> byteWriters = new ThreadLocal<ByteWriter>() {
protected ByteWriter initialValue() {
return new ByteWriter(encoderFactory.getEncoder(), bufferSize);
}
};
public ByteWriter getByteWriter(java.io.OutputStream outputStream) {
ByteWriter ret = new ByteWriter(encoderFactory.getEncoder(), reentrantBufferSize);
return ret.init(outputStream);
}
public void setEncoderFactory(EncoderFactory encoderFactory) {
if (encoderFactory == null) {
throw new IllegalArgumentException("encoderFactory can not be null");
}
this.encoderFactory = encoderFactory;
}
}
问题2
- 使用了
jfinal
自带的验证码类缓存CaptchaCache
就会报这个错. 它是通过CaptchaManager.getCaptchaCache()
获取的。-
报错提示说它创建了一个线程,查看它的源码,发现它在内部创建了一个定时器
Timer
,而Timer
就是一个线程。 -
这个定时器不是一次性的,不会自己关闭,所以还需要手动关闭这个定时器。
-
timer
属性是private
的,因此也只能通过反射获取这个属性值。 -
可以通过子类化重载
removeAll
方法来获取并关闭timer
。
-
CaptchaCache.java
public class CaptchaCache implements ICaptchaCache {
private ConcurrentHashMap<String, Captcha> map = new ConcurrentHashMap<String, Captcha>();
private int interval = 90 * 1000; // timer 调度间隔为 90 秒
private Timer timer;
public CaptchaCache() {
autoRemoveExpiredCaptcha();
}
/**
* 定期移除过期的验证码
*/
private void autoRemoveExpiredCaptcha() {
timer = new Timer("CaptchaCache", true);
timer.schedule(
new TimerTask() {
public void run() {
for (Entry<String, Captcha> e : map.entrySet()) {
if (e.getValue().isExpired()) {
map.remove(e.getKey());
}
}
}
},
interval,
interval
);
}
...
MyCaptchaCache.java
import com.jfinal.captcha.CaptchaCache;
import java.lang.reflect.Field;
import java.util.Timer;
public class MyCaptchaCache extends CaptchaCache {
public void removeAll() {
super.removeAll();
Field f= null;
try {
f = getClass().getSuperclass().getDeclaredField("timer");
f.setAccessible(true);
Timer timer = (Timer) f.get(this);
if(timer != null)
timer.cancel();
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}catch (Exception e){
e.printStackTrace();
}
}
}
如何使用?
- 可以在配置类
JFinalConfig
的子类DemoConfig
里使用,用两个新类的实例替换默认的实例。
MyWriterBuffer writerBuffer = new MyWriterBuffer();
MyCaptchaCache captchaCache = new MyCaptchaCache();
public void configEngine(Engine me) {
...
me.getEngineConfig().setWriterBuffer(writerBuffer);
}
public void configConstant(Constants me) {
...
me.setCaptchaCache(captchaCache);
}
@Override
public void onStop() {
// 异步任务线程池
//ThreadUtils.get().getExecutor().shutdown();
// 数据库连接池
//DemoConfig.stopDruid();
//BUG: 1. 貌似Jfinal没有释放JDBCDriver.
//clearJDBCDriver();
CaptchaManager.me().getCaptchaCache().removeAll();
// 日志服务
//LogManager.shutdown();
}
其他
-
注意: 如果还有其他定时任务,线程池,
Druid
数据库连接池或其他线程本地变量也应该在onStop()
方法清除掉。 -
在
linux
上, 可能会使用ubic stop tomcat8
工具来关闭tomcat
。这时候如果不处理好这些内存泄露的问题会关闭不了,造成网站已关闭的假象,导致如果再次ubic start tomcat8
启动网站会报端口占用错误。