7

在我的 java web 应用程序中,我有一个后台工作线程,它需要大量的堆栈空间,因为它使用 activiti 工作流引擎和 groovy 脚本任务运行一个非常复杂的工作流。

目前我需要在 64 位 Java 和 Tomcat 上将 JVM Xss 设置设置为 16MB,以规避任何 StackOverflowErrors。如果发生错误,堆栈跟踪真的很大(几百行长),但它都发生在引擎内部,所以我真的无能为力。

现在我的问题是:有没有办法在运行时增加单个线程的堆栈大小?

我想将应用程序中所有线程的 JVM 默认 Xss 设置降低到 512k,我知道这已经足够了,并且只运行 16M 的工作线程。

Java API 为 Thread 类的构造函数提供了有关此主题的一些信息:

public Thread(ThreadGroup group,
              Runnable target,
              String name,
              long stackSize)

但它提到,不能保证这种行为([1]),我没有找到任何信息,如果它可以在 Windows 上工作。

另外,如果一个线程的堆栈空间不能增加,我必须指定16MB作为默认值,那么设置这么高会有什么后果呢?这是否意味着每个新线程在初始化时都会保留 16MB 的内存(即 200 个线程 * 16MB:3.2 GB)?

据我从 jconsole 和 taskmgr 可以看出,自 Xss 设置增加以来,内存占用似乎没有太大变化,但也许我遗漏了一些东西。

任何帮助或澄清将不胜感激。

[1]:http ://docs.oracle.com/javase/6/docs/api/java/lang/Thread.html#Thread (java.lang.ThreadGroup, java.lang.Runnable, java.lang.String,长)

4

2 回答 2

4

在 HotSpot 上指定最大尺寸而不是-Xss它将使用的尺寸毫无价值。使用的大小基于使用情况,因此如果您指定这个不合理的大,它将浪费虚拟内存(这在 32 位 JVM 上可能是一个问题),但不会浪费物理内存。如果您有一个 64 位 JVM,那么将任何线程设为您想要的最大线程并没有什么缺点。

另外,如果一个线程的堆栈空间不能增加,我必须指定16MB作为默认值,那么设置这么高会有什么后果呢?这是否意味着每个新线程在初始化时都会保留 16MB 的内存(即 200 个线程 * 16MB:3.2 GB)?

每个线程将使用这么多虚拟内存。在 32 位 JVM 上,即使您实际上使用的内存很少,也会用完地址空间。在 64 位 JVM 上,您的限制在 TB 中,只有实际使用的堆栈才会使用主内存。

于 2012-07-16T15:44:45.110 回答
0

如果相关线程是进程密集型的。将此线程的逻辑实现为另一个应用程序,并使用队列传递信息和流程变量。您可以根据需要增加此应用程序的内存参数。

从文档链接:

虚拟机可以自由地将 stackSize 参数视​​为建议。如果平台的指定值过低,虚拟机可能会使用一些特定于平台的最小值;如果指定的值过高,则虚拟机可能会改为使用某些特定于平台的最大值。同样,虚拟机可以随意向上或向下舍入指定的值(或完全忽略它)。

于 2012-07-16T15:13:14.700 回答