28

(危险风格的问题,我希望当我遇到这个问题时答案已经在线)

使用 Java 1.4,我有一个方法我想在某些时候作为线程运行,但在其他时候不想。所以我将它声明为 Thread 的子类,然后根据我的需要调用 start() 或 run()。

但是我发现我的程序会随着时间的推移而泄漏内存。我究竟做错了什么?

4

3 回答 3

48

这是 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();
于 2008-09-20T10:25:06.870 回答
3

我怀疑构造 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 的实例是垃圾收集的。

于 2008-09-20T11:26:57.223 回答
2

让我们看看我们是否可以更接近问题的核心:

如果您使用 start() 启动程序(假设)1000 x,然后在线程中使用 run() 启动 1000 x,是否都会释放内存?如果是这样,那么您的算法应该被检查(即外部对象,如您的 Runnable 中使用的向量)。

如果没有如上所述的内存泄漏,那么您应该调查有关 JVM 的启动参数和线程的内存使用情况。

于 2008-09-20T10:26:17.507 回答