现象
正常运行一年的系统,突然不能上传文件了。
调查
查看日志,提示磁盘空间不足了。
java.io.IOException: 磁盘空间不足。
at java.io.RandomAccessFile.writeBytes(Native Method) ~[na:1.8.0_231]
at java.io.RandomAccessFile.write(Unknown Source) ~[na:1.8.0_231]
at javax.imageio.stream.FileCacheImageOutputStream.write(Unknown Source) ~[na:1.8.0_231]
at com.sun.imageio.plugins.jpeg.JPEGImageWriter.writeOutputData(Unknown Source) ~[na:1.8.0_231]
at com.sun.imageio.plugins.jpeg.JPEGImageWriter.writeImage(Native Method) ~[na:1.8.0_231]
at com.sun.imageio.plugins.jpeg.JPEGImageWriter.writeOnThread(Unknown Source) ~[na:1.8.0_231]
at com.sun.imageio.plugins.jpeg.JPEGImageWriter.write(Unknown Source) ~[na:1.8.0_231]
at javax.imageio.ImageWriter.write(Unknown Source) ~[na:1.8.0_231]
at javax.imageio.ImageIO.doWrite(Unknown Source) ~[na:1.8.0_231]
at javax.imageio.ImageIO.write(Unknown Source) ~[na:1.8.0_231]
此系统只有WEB服务,没有任何增量的存储,怎么会把磁盘空间占满呢?
查看后,发现大量文件存在了C:\Program Files\Apache Software Foundation\Tomcat 8.0\work\Catalina\localhost目录下。
有大量的uploadxxxx.tmp文件。
原因
系统使用了org.springframework.web.multipart.commons.CommonsMultipartResolver理文件上传。
当收到请求时 DispatcherServlet 的 checkMultipart() 方法会调用 MultipartResolver 的 isMultipart() 方法判断请求中是否包含文件。如果请求数据中包含文件,则调用 MultipartResolver 的 resolveMultipart() 方法对请求的数据进行解析,然后将文件数据解析成 MultipartFile 并封装在 MultipartHttpServletRequest (继承了 HttpServletRequest) 对象中,最后传递给 Controller,
MultipartFile 封装了请求数据中的文件,此时这个文件存储在内存中或临时的磁盘文件中,需要将其转存到一个合适的位置。
<!-- 定义文件上传解析器 -->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!-- 设定默认编码 -->
<property name="defaultEncoding" value="UTF-8"></property>
<!-- 设定文件上传的最大值为5MB,5*1024*1024 -->
<property name="maxUploadSize" value="5242880"></property>
<!-- 设定文件上传时写入内存的最大值,如果小于这个参数不会生成临时文件,默认为10240 -->
<property name="maxInMemorySize" value="40960"></property>
<!-- 上传文件的临时路径 -->
<property name="uploadTempDir" value="fileUpload/temp"></property>
<!-- 延迟文件解析 -->
<property name="resolveLazily" value="true"/>
</bean>