Java 拒绝启动 - 无法为对象堆保留足够的空间

标签 java linux memory jvm heap-memory

背景

我们有大约 20 个 linux Blade 库。有些运行 Suse,有些运行 Redhat。 ALL 共享 NAS 空间,包含以下 3 个文件夹:

  • /NAS/app/java - 指向 Java JDK 安装的符号链接(symbolic link)。当前版本 1.5.0_10
  • /NAS/app/lib - 指向我们应用程序版本的符号链接(symbolic link)。
  • /NAS/data - 写入输出的目录

我们所有的机器都有 2 个处理器(超线程),具有 4gb 的物理内存和 4gb 的交换空间。我们将每台机器在给定时间可以处理的“作业”数量限制为 6 个(这个数字可能需要更改,但这不属于当前问题,因此请暂时忽略它)。

我们的一些作业将最大堆大小设置为 512mb,而另一些作业将最大堆大小保留为 2048mb。同样,我们意识到如果在同一台机器上启动 6 个作业并将堆大小设置为 2048,我们可能会超出可用内存,但据我们所知,这还没有发生。

问题

有时,Job 会立即失败并显示以下消息:

Error occurred during initialization of VM
Could not reserve enough space for object heap
Could not create the Java virtual machine.

我们过去常常将此归咎于同一台机器上同时运行的作业太多。该问题很少发生(可能每月一次),我们只需重新启动它,一切都会好起来的。

这个问题最近变得更糟了。我们所有请求最大堆大小为 2048m 的作业几乎每次都会立即失败,并且需要重新启动几次才能完成。

我们已经使用了个别机器并尝试手动执行它们,结果相同。

调试

事实证明,问题只存在于我们的 SuSE 盒子。它发生得更频繁的原因是因为我们一直在添加更多的机器,而新的机器是 SuSE。

SuSE 框中的“cat/proc/version”给我们:

Linux version 2.6.5-7.244-bigsmp (geeko@buildhost) (gcc version 3.3.3 (SuSE Linux)) #1 SMP Mon Dec 12 18:32:25 UTC 2005

RedHat 盒子上的“cat/proc/version”给我们:

Linux version 2.4.21-32.0.1.ELsmp (bhcompile@bugs.build.redhat.com) (gcc version 3.2.3 20030502 (Red Hat Linux 3.2.3-52)) #1 SMP Tue May 17 17:52:23 EDT 2005

'uname -a' 在两种类型的机器上为我们提供以下信息:

UTC 2005 i686 i686 i386 GNU/Linux

机器上没有正在运行的作业,也没有其他进程占用太多内存。当前运行的所有进程可能总共使用 100mb。

'top' 当前显示如下:

Mem:   4146528k total,  3536360k used,   610168k free,   132136k buffers
Swap:  4194288k total,        0k used,  4194288k free,  3283908k cached

'vmstat' 当前显示如下:

procs -----------memory---------- ---swap-- -----io---- --system-- ----cpu----
r  b   swpd   free   buff  cache   si   so    bi    bo   in    cs us sy id wa
0  0      0 610292 132136 3283908    0    0     0     2   26    15  0  0 100  0

如果我们使用以下命令行启 Action 业(最大堆为 1850mb),它会正常启动:

java/bin/java -Xmx1850M -cp helloworld.jar HelloWorld
Hello World

如果我们将最大堆大小增加到 1875mb,它会失败:

java/bin/java -Xmx1875M -cp helloworld.jar HelloWorld
Error occurred during initialization of VM
Could not reserve enough space for object heap
Could not create the Java virtual machine.

很明显,当前正在使用的内存是用于缓冲/缓存的,这就是为什么很少显示为“空闲”的原因。不清楚的是为什么有一个神奇的 1850mb 行,任何更高的值都意味着 Java 无法启动。

任何解释将不胜感激。

最佳答案

您使用的是 32 位操作系统,因此您会看到总大小的限制。其他答案已经更详细地介绍了这一点,因此我将避免重复他们的信息。

我最近在我们的服务器上注意到的一个行为是,使用 -Xmx 指定最大堆大小而不使用 -Xms 指定最小堆大小会导致 Java服务器 VM 立即尝试分配最大堆大小所需的所有内存。当然,如果应用程序达到该堆大小,这就是您需要的内存量。但很有可能,您的应用一开始会使用相对较小的堆,可能在以后需要更大的堆。此外,指定最小堆大小将使您的应用从较小的堆开始,然后逐渐增大该堆。

所有这些都不会帮助您增加最大堆大小,但我认为它可能会有所帮助,所以...

关于Java 拒绝启动 - 无法为对象堆保留足够的空间,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1058471/

相关文章:

java - 如何使用java在android中绘制带有圆角矩形的二维码?

Java 在终止之前等待所有线程

linux - 如何不使用正则表达式和 sed 检测结尾 '/'?

linux - bash中十六进制到十进制的转换

c++ - 如何使用多指针写入 ProcessMemory

java - 我的应用程序的调试构建工作正常但不是发布构建签名版本

java - double 和 Double 比较的区别

linux - 如何从 grep -R 中排除目录?

linux - 当我使用 brk 系统调用时,我分配的内存实际上从哪里开始

c++ - 访问冲突读取位置 0xcdcdcdcd。 C++