(危险风格的问题,我希望当我遇到这个问题时答案已经在线)
使用 Java 1.4,我有一个方法我想在某些时候作为线程运行,但在其他时候不想。所以我将它声明为 Thread 的子类,然后根据我的需要调用 start() 或 run()。
但是我发现我的程序会随着时间的推移而泄漏内存。我究竟做错了什么?
(危险风格的问题,我希望当我遇到这个问题时答案已经在线)
使用 Java 1.4,我有一个方法我想在某些时候作为线程运行,但在其他时候不想。所以我将它声明为 Thread 的子类,然后根据我的需要调用 start() 或 run()。
但是我发现我的程序会随着时间的推移而泄漏内存。我究竟做错了什么?
这是 Java 1.4 中的一个已知错误: http://bugs.sun.com/bugdatabase/view_bug.do;jsessionid=5869e03fee226ffffffffc40d4fa881a86e3:WuuT?bug_id= 4533087
它已在 Java 1.5 中修复,但 Sun 不打算在 1.4 中修复它。
问题是,在构造时,aThread
被添加到内螺纹表中的引用列表中。在它的 start() 方法完成之前,它不会从该列表中删除。只要该引用存在,它就不会被垃圾收集。
所以,除非你确定要调用它的start()
方法,否则永远不要创建一个线程。不应直接调用Thread
对象的方法。run()
更好的编码方式是实现Runnable
接口而不是子类Thread
。当你不需要线程时,调用
myRunnable.run();
当您确实需要线程时:
Thread myThread = new Thread(myRunnable);
myThread.start();
我怀疑构造 Thread 的实例或其子类会泄漏内存。首先,Javadocs 或 Java 语言规范中没有提到任何类型的东西。其次,我进行了一个简单的测试,它还显示没有内存泄漏(至少在 32 位 x86 Linux 2.6 上的 Sun 的 JDK 1.5.0_05 上没有):
public final class Test {
public static final void main(String[] params) throws Exception {
final Runtime rt = Runtime.getRuntime();
long i = 0;
while(true) {
new MyThread().run();
i++;
if ((i % 100) == 0) {
System.out.println((i / 100) + ": " + (rt.freeMemory() / 1024 / 1024) + " " + (rt.totalMemory() / 1024 / 1024));
}
}
}
static class MyThread extends Thread {
private final byte[] tmp = new byte[10 * 1024 * 1024];
public void run() {
System.out.print(".");
}
}
}
编辑:只是总结一下上面测试的想法。Thread 的 MyThread 子类的每个实例都引用它自己的 10 MB 数组。如果 MyThread 的实例没有被垃圾回收,JVM 会很快耗尽内存。但是,运行测试代码表明 JVM 使用的内存量很小,与目前构建的 MyThreads 数量无关。我声称这是因为 MyThread 的实例是垃圾收集的。
让我们看看我们是否可以更接近问题的核心:
如果您使用 start() 启动程序(假设)1000 x,然后在线程中使用 run() 启动 1000 x,是否都会释放内存?如果是这样,那么您的算法应该被检查(即外部对象,如您的 Runnable 中使用的向量)。
如果没有如上所述的内存泄漏,那么您应该调查有关 JVM 的启动参数和线程的内存使用情况。