0

我有一个在 Tomcat 7 上运行的 Servlet 应用程序。我在 setenv.sh 中设置了以下内容

CATALINA_OPTS="
-server 
-Xms1G 
-Xmx5G
-XX:MaxPermSize=512m
-Dfile.encoding=UTF-8";

该服务器在 Ubuntu 12.04LTS 上运行,具有 6 个内核和 8GB 内存。我的应用程序是一个 Grails/Spring/Java 应用程序,用户可以在其中上传图像。有时会出现 3-5 个用户同时开始上传图像。在这些情况下,我的 Tomcat 由于以下错误而崩溃:

java.lang.OutOfMemoryError: Java heap space

我知道我必须增加我的 Tomcat 的 Xmx 参数来防止这个问题。但说真的,我的情况有什么问题吗?

  1. 是否有像我的场景一样处理文件上传的最佳实践?

  2. 我可以如何防止这个问题吗?我的意思是当 5GB 不足以进行 3-5 次并行上传时,我需要多少资源来进行数百次并行上传?

  3. 我的应用程序如此糟糕还是文件上传占用如此多资源是否正常?

注意:用户上传的文件为 2-8 MB。

这就是我在我看来上传的方式:

   <g:form method="post" action="save" enctype="multipart/form-data">
    <input type="file" name="file"/>
    <input type="submit"/>
   </g:form>

在我的控制器中:

def file = request.getFile('image')
byte [] byteFile = file.getBytes()

在 config/spring/resource.groovy 我做了:

beans = { 
  multipartResolver(org.springframework.web.multipart.commons.CommonsMultipartResolver){

        // Max in memory 100kbytes
        maxInMemorySize=102400

    }

} 

上传的文件是图片。它们将被缩放并保存到磁盘。

图像将保存为:

bytes [] currentImage = // some image in bytes ...
def newFile = new FileOutputStream(fullPath)
newFile.write(currentImage)
newFile.close()
4

1 回答 1

1

避免MultipartFile.getBytes()MultipartFile.getInputStream()改为使用以避免将整个文件内容读入内存。

因此,将您的代码更改为:

def file = request.getFile('image')
InputStream inputStream = file.inputStream

你没有提到你对上传的文件做了什么,但如果你想最小化堆使用,请确保在你传递的应用程序的整个层中InputStream不要转换为 a 。byte[]

于 2013-08-26T13:00:48.780 回答