Java修改图片尺寸,总是报内存溢出怎么解决?

项目需求:有很多尺寸很大的图片(图片大小可能几十MB,甚至上百MB),需要等比例缩小它们的尺寸,相当于生成缩略图。

例如:原图8268x6201,需要压缩成400x300。

我现在的方法:将图片整个读入内存,然后修改宽、高,最后写回磁盘。

核心代码大概这样:

BufferedImage oldImg = ImageIO.read(in);
BufferedImage newImg = new BufferedImage(400, 300, BufferedImage.TYPE_INT_RGB);
Graphs2D g = newImg.createGraphics();
g.drawImage(oldImg, 0, 0, 400, 300, null);
g.dispose();
ImageIO.write(newImg, 'jpg', new BufferedStream(out));

但是由于图片本身很大,而且可能会同时有多个线程在转,所以频繁发生内存溢出,报错:

java.lang.OutOfMemoryError: Java heap space.

尝试:

  1. 我试过增大分配给JVM的内存,但是效果不明显。首先机器内存有限,不可能无限增加。而且图片的大小没有上限,所以此方法不能很好地解决问题。

  2. 理想的情况,应该是通过一种“流式的算法”,一边把原始图片从磁盘读到内存,一边修改尺寸,一边写到磁盘生成目标图片。这种算法理论上只需要常数级内存空间。不过我在网上找了很久也没有找到。

我对图形学、图片格式、图片处理相关的知识不熟悉,不知道是否存在这样的算法,以及Java是否支持这样的算法。在此请教大家。


慕田峪7331174
浏览 1382回答 5
5回答

斯蒂芬大帝

GraphicsMagick+im4java可以处理,GraphicsMagick不用把整张图片读取到内存中,比你用原生的效率高多了,你可以搜索看看,以前我们就是用GraphicsMagick处理图片裁剪的

慕森王

可以试试 ImageMagick

青春有我

思路可以换一换,显然内存中是能存储压缩后的图像的,而大图像应该应该读一部分处理后放到压缩后的图形中,总得来说这个问题不算难,思路换一下就行了
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Java