我的目标是在抛出 java.lang.OutOfMemoryError 时执行脚本(Windows 批处理文件),按照Hotspot VM Options中描述的选项
-XX:OnOutOfMemoryError="<cmd args>; <cmd args>"
当抛出 OutOfMemoryError 时,我无法执行脚本或任何其他进程。首先我尝试启动 cmd.exe 并指定脚本的路径,我修改了双引号的位置,使用前斜杠而不是反斜杠,使用双反斜杠。我完全放弃了斜杠和引号,并在工作目录或 PATH 上尝试了简单的 .exe。没有任何效果。
我什至无法尝试运行任何可执行文件。通过使用ProcMon,我确定 JVM 甚至没有在寻找具有我指定名称的可执行文件,更不用说尝试启动它了。
我知道我在 JVM 参数中正确指定了该选项,因为如果我故意插入错字,它会抱怨。我正在尝试以下操作:
java -XX:OnOutOfMemoryError=myprocess.exe com.mycompany.SimulateOome
java -XX:OnOutOfMemoryError="myprocess.exe" com.mycompany.SimulateOome
java "-XX:OnOutOfMemoryError=myprocess.exe" com.mycompany.SimulateOome
这会产生到 stdout/err 的预期堆栈跟踪。
这可能是我的测试课。我当前的版本只是抛出 java.lang.OutOfMemoryError。我还通过分配一个大数组来模拟一个更有机的 OOME。两者都不起作用。
我在 Windows 2003/2008 上使用 JDK 1.6.0_38,但也尝试了 JDK 1.7。
编辑:
测试班有问题。简单地抛出异常或简单的大数组分配都不会导致 OOME 处理程序触发,尽管否则会成功抛出 OOME。有关触发 OOME 处理程序的 Perception 的 OutOfMem 类,请参见下文。整理好之后,语法就很简单了,引号可以包围整个选项,也可以只是等号后面的部分。对于 Windows shell 脚本,可以使用:
java "-XX:OnOutOfMemoryError=c:\windows\system32\cmd.exe /c c:\scripts\my_oome_handler.bat" com.mycompany.MyClass
或者
java -XX:OnOutOfMemoryError="c:\windows\system32\cmd.exe /c c:\scripts\my_oome_handler.bat" com.mycompany.MyClass