一. 问题描述
JBoss 4.0.5GA 在若干次部署或重新部署应用程序(war、ear 等)后会马上出现 PermGen Memory Leak : Out of memory exception 内存溢出异常。
参阅:http://wiki.jboss.org/wiki/Wiki.jsp?page=OutOfMemoryExceptionWhenRedeploying
二. 问题分析
1. 猜测:应用程序的类中包括了 static 成员定义,当重新部署时 JBoss 使用新的 Class Loader 载入应用程序的所有类。由于 static 成员是置于 JBoss 的固定内存单元,多次部署后占用内存空间以倍数递增,最终使该部分内存满载,进而在下一次部署或重新部署时导致内存溢出异常。
参阅:http://jboss.org/index.html?module=bb&op=viewtopic&p=3918780
讨论标题:Multiple redeployment gets PermGen error
The permanent generation holds class-level data (that is, any fields defined as static). My guess that your classes contain static data, and when you redeploy, the redeployed classes are loaded by a new class loader and thus the static data from the old classes and the new classes are now taking up room. Do this several times and the permanent generation will fill up. If my guess is correct, the solution would be to modify your code to not use static data in any of your classes. Of course you should also check any class libraries on which your code depends (jar files you package with your ear) for the same thing. But then I could be wrong.
2. 在 Bug 2299 报告的讨论中,涉及到关于 Struts 框架中引用了 BeanUtils 1.6 的讨论。
三. 参阅
1. 相关 Bug 报告:
1)Bug 1319
http://jira.jboss.com/jira/browse/JBAS-1319
(声称到 4.0.3RC1 已解决)
2)Bug 2299
http://jira.jboss.com/jira/browse/JBAS-2299
(声称到 4.2.0 才会解决)
2. 关于 Permanent Generation Heap 的概念以及如何在 JVM 启动命令行中添加重新定义 PermSize 和 MaxPermSize 的参数:
http://wiki.jboss.org/wiki/Wiki.jsp?page=PermanentGeneration
3. 关于如何查看 JBoss 内存使用状况的操作指引:
http://wiki.jboss.org/wiki/Wiki.jsp?page=HowCanIDisplayTheMemoryPoolUsage
1)打开 http://server-name-or-ip/jmx-console/HtmlAdaptor
2)在 jboss.system 节点找到 type=ServerInfo ,点击进入
3)找到 java.lang.String listMemoryPools() 方法,点击 Invoke 按钮执行
PermGen 内存状况在最下面一行,当已使用的内存数量逼近最大可用内存限制时应考虑重启 JBoss 。
四. 建议
在可能的条件下使用更稳定但不是免费的 IBM Websphere Application Server 代替 JBoss Application Server 。根据外间性能评测,这两者在性能上比较接近。
性能评测参阅: http://www.51testing.com/html/27/1132.html
但如果应用程序是在正式运营期间,由于可能的重新部署情况比开发和测试阶段少很多,因此不需过多担心。 使用以下方案可减少或避免内存溢出异常的出现:
1. 部署或重新部署应用程序后通过 jmx-console 检查内存占用状况,在 PermGen Size 逼近限制值时及时重启 JBoss AS 。
2. 增大 PermSize 和 MaxPermSize (不建议使用相同的值)。