强制JVM释放本地内存。

7
我偶尔在Eclipse中运行需要大量内存的任务。因此,当任务正在运行时,JVM会吞噬约2-3GB的RAM,这是可以接受的。但是,一旦JVM占用了该内存,它就不会释放它,我会遇到这样一种情况:堆中使用的内存约为200MB,总堆大小约为3GB,这真的是不希望看到的,因为其他程序需要内存。
我尝试使用Max/MinHeapFreeRatio参数来强制JVM减少未使用内存的消耗。这是我的Eclipse config.ini文件:
-startup
plugins/org.eclipse.equinox.launcher_1.2.0.v20110502.jar
--launcher.library
plugins/org.eclipse.equinox.launcher.win32.win32.x86_64_1.1.100.v20110502
-vm
c:/Program Files/Java/jdk1.6.0_26/bin/javaw.exe
-showlocation
-product
org.eclipse.epp.package.jee.product
--launcher.defaultAction
openFile
--launcher.XXMaxPermSize
256M
-showsplash
org.eclipse.platform
--launcher.XXMaxPermSize
256m
--launcher.defaultAction
openFile
-vmargs
-Duser.name=Michael Nesterenko
-Dosgi.requiredJavaVersion=1.5
-Xms512m
-Xmx4096m
-XX:MinHeapFreeRatio=10
-XX:MaxHeapFreeRatio=30

但这并没有帮助,我仍然会遇到大量未使用的内存的情况。

有两个不同的快捷方式/ shell 脚本来启动 Eclipse,对于需要执行内存密集型任务时使用不同的参数集,这样做是否实用? - matt freake
1个回答

4

Java在启动时分配最大堆大小作为虚拟内存。随着使用内存,它被分配为实际的主内存。它从不缩小这个大小。释放这个内存的唯一方法是程序退出。

Java将压缩其内存使用,因此它使用过的许多页面可以交换到磁盘,但如果需要再次使用它们,则可能会产生显着的性能影响。


5
我不同意。内存是可以收缩的。根据这篇文档(http://www.oracle.com/technetwork/java/javase/tech/vmoptions-jsp-140102.html),设置`-XX:MaxHeapFreeRatio=70`可以避免内存缩减,在垃圾回收后堆中有70%以上的空闲空间时,堆将不会继续增长。此外,根据这份错误报告(http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=6498735):如果Java虚拟机中没有活动对象需要使用内存,HotSpot JVM会将内存释放回操作系统。 - alain.janinm
1
堆使用量会缩小,但应用程序的常驻大小不会。我也不知道错误审阅者在说什么。我从来没有看到虚拟内存占用量会缩小。 - Peter Lawrey
1
我发现JVM应该进行的优化和实际情况经常存在差异。一个例子是逃逸分析。理论上,它可以消除很多对象创建,但在现实中很难找到这种情况的例子。 - Peter Lawrey
@PeterLawrey 好的,你是在Windows上还是只有一个核心?我使用Ubuntu带有4个核心,使用的内存大约是你的4倍:)。我为一份报告制作了这个,并且从结果中绘制了图形(使用Debian 4核心),你可以在Github上看到其中一个。你关于任务管理器的评论是正确的,我以前从未注意过...内存值保持不变,我不知道我们是否可以依赖它。如果Java进程不释放内存,未提交的内存去哪里了?我猜答案可能取决于JVM实现和操作系统。在这里我缺乏知识,抱歉... - alain.janinm
1
@alain.janinm 这是一台拥有8个核心和32GB内存(约23GB可用)的Windows 7电脑。我想,由于空闲内存较高,操作系统可能不会回收内存,因为它并不需要。 - Peter Lawrey
显示剩余5条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接