performance - grails 2.0.3 中 byte[] 的内存泄漏

标签 performance tomcat grails file-upload memory-leaks

所以我有一个应用程序,允许您上传文档和图像并将它们保存为数据库中的 byte[]。

(我创建了一个最小的示例项目,您可以在此处以 zip 形式下载并导入到 STS http://dl.dropbox.com/u/2342474/byteMemoryLeak.zip ...您所要做的就是转到 datasource.groovy 并更改数据库设置)

所以域对象看起来像这样:

class Test { 

    byte[] data 

    static constraints = { 
        data nullable: false, maxSize: 1024*1024*40 
    } 
} 

Controller 和 View 都是直接使用generate-all命令生成的...这里没有任何改变...

现在,当我们启动项目时,我们会转到测试 Controller ...点击创建新...上传文档...使用 5 - 10 mb 之类的内容...然后点击创建...

我预计在执行 save() 方法以及调用 show 方法时内存使用量会增加...但是一旦这些对象加载完成,GC 应该拾取这些对象并将它们扔掉,因为它们是目前不需要...现在我使用 Your Kit Java Profiler 来检查内存使用情况,我可以看到我的 byte[] 有 3 个引用,它们具有弱引用或无法访问...为什么会这样?当 show() 方法或 save() 方法完成时,它们不应该被删除吗?

这里是屏幕截图:[内存泄漏]:/image/R7KMt.png

我是否需要对 tomcat 进行特殊设置才能更频繁地运行 gc?

我对此感到非常困惑,它实际上会导致我现在正在测试的应用程序出现问题,因为人们正在向其中上传大量文档,而堆内存会变得越来越满,并且几乎从未被清理过,所以我在某个时候遇到内存不足异常...实际上在 list() 方法中也发生了同样的情况,它加载了所有对象,现在实际上对于每个对象, byte[] 已加载,现在在内存中并且没有得到清理干净了...

也许我只是错过了一些东西或做错了一些事情...任何帮助将不胜感激...

谢谢 克里斯

最佳答案

Grails 应用程序通常使用在服务器模式下创建的 JRE 运行,这会影响内存的工作方式。基本上,当在“客户端”模式下运行时,它会尝试使用更少的内存并更频繁地释放它,而当在服务器模式下运行时,JRE 会尝试使用更多的内存来提供更高的速度。您可以在这里阅读一些讨论:Real differences between “java -server” and “java -client”

基本上,JRE 会允许堆增长到其极限,然后才会考虑释放内存。

我下载了你的示例并启动了它,然后我多次上传了相同的文件。 java进程的内存使用量增长如下(每个请求:822mb -> 917mb -> 1.03mb,然后停止。接下来的上传没有增加内存使用量,也没有抛出PermGen space异常。

所以,我认为这里没有发生内存泄漏。您遇到的只是 JRE 尝试在仍然可以分配新内存之前不运行 GC(因为通常,运行 GC 比分配新内存要慢)。

希望这有帮助。

关于performance - grails 2.0.3 中 byte[] 的内存泄漏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10341664/

相关文章:

tomcat - Pivotal 不显示来自 STDERR 的日志

tomcat - 两个 tomcat 服务器使用相同的端口,为 iis 重写规则

search - Solr 不会搜索属于嵌套实体的字段

grails - 如何在grails中保存对象

c# - 确定线程性能

java - 哪个更快?

c# - 我该怎么做才能使这个循环运行得更快?

sql - 使用OR语句驯服MySQL查询性能时出现问题

grails - 在同一张 table 上造成多对多关系

grails - 使用Grails 3将响应从长时间运行的流程返回给客户端