0

我有一个 tomcat webapp,我在其中使用 java 执行器来执行一些可运行文件。

首先,我从我的执行程序实例中分配执行程序服务,如下所示:

executorService = Executors.newFixedThreadPool(nThreads, new MyThreadFactory(threadFactoryName))

然后,我使用执行器服务启动任务:

executorService.execute(new MyRunnable);

然后,它似乎工作。但我的问题是,我正在追逐某种泄漏,在运行服务器一段时间后出现此错误:

Caused by: java.lang.OutOfMemoryError: unable to create new native thread
at java.lang.Thread.start0(Native Method)
at java.lang.Thread.start(Thread.java:640)
at java.util.concurrent.ThreadPoolExecutor.addIfUnderCorePoolSize(ThreadPoolExecutor.java:703)
at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:652)
at something.server.ThreadExecutor.execute(MyRunnable.java:91)

我在 tomcat 实例上使用了 java VisualVM 来跟踪线程分配。我看到当我调用“newFixedThreadPool”时,我看到了“nThreads”新线程。但是在他们的工作完成后,我看到线程的状态是“等待”。

这正常吗?我不相信,我知道可运行对象完成了他们的工作,但执行程序服务从不释放线程。我能做些什么来释放它们,还是我完全超出了范围?

4

3 回答 3

4

ExecutorService我认为您正在为每个请求或类似的东西实例化一个新的。

您应该只实例化一个ExecutorService并重用它,或者shutdown()在完成提交任务后调用该服务。该shutdown函数将等待任务完成,然后释放线程。

于 2012-12-13T22:15:09.493 回答
2

这正常吗?我不相信,我知道可运行对象完成了他们的工作,但执行程序服务从不释放线程。

是的,这很正常。实际上,这种行为是可取的,因为创建新线程是一项昂贵的操作。池线程(或至少“核心”池线程)将继续存在,直到执行器关闭。

我能做些什么来释放它们,还是我完全超出了范围?

您应该能够将线程池配置为具有更小的“核心”线程池和/或更小的 keepAlive 时间。有关详细信息,请参阅javadoc


[M] 我的问题是,我正在追逐某种泄漏,在运行服务器一段时间后出现此错误:

OOME 不一定表示泄漏。这可能表明您没有为尝试创建的线程数留下足够的非堆内存。(或者换句话说,你的线程池太大了。)

于 2012-12-13T22:13:32.537 回答
2

您正在实例化一个固定的线程池。在您通过调用orExecutorService终止自身之前,不会释放线程。ExecutorServiceExecutorService.shutdown()ExecutorService.shutdownNow()

于 2012-12-13T22:15:02.737 回答