0

我已经读取了相当大的图像(jpegs)的应用程序,但只需要处理较小的图像,所以我用类似的东西对它们进行二次采样

ImageReadParam param = reader.getDefaultReadParam();
param.setSourceSubsampling(4, 4, 0, 0);
img = reader.read(0);

但是,由于 jpeg 阅读器中的一个错误不能处理一些我已经回退到其他方法的元数据,一个是使用 JAI 读取图像然后调整大小(代码在下面,不是我必须使用反射作为一些部署环境没有可用的 JAI,我知道我可以更好地围绕这个进行设计,但就是这样)。

try {
Class<?> c = ImageUtil.class.getClassLoader().loadClass("javax.media.jai.JAI");

if (c != null) { 
    URL url = new URL("file://" + file.getAbsolutePath());
    Method m = c.getMethod("create", String.class, Object.class);
    Object pi = m.invoke(null, "url", url);
    img = (BufferedImage) pi.getClass().getMethod("getAsBufferedImage").invoke(pi);
}
} catch (Throwable tt) {
// ...
}

然而,有些图像真的很大,有时我会出现内存不足的异常,无论如何我可以让 JAI 在以我使用 ImageReader 读取图像的方式读取图像时对图像进行二次采样?

4

2 回答 2

2
RenderedOp rop = JAI.create("fileload", file.getAbsolutePath());

ParameterBlock pb = new ParameterBlock();
pb.addSource(rop);
pb.add(0.5f);
pb.add(0.5f);
rop = JAI.create("scale", pb);

// For better looking results, but slower:
// rop = JAI.create("SubsampleAverage", pb);

BufferedImage bufImg = rop.getAsBufferedImage();   
于 2012-11-21T15:11:15.873 回答
0

我假设当您尝试转换为 BufferedImage 时会引发内存不足异常?

如果是这种情况,那么也许考虑坚持使用 JAI 的 create 方法返回的 RenderedOp,而是使用 ParameterBlock 和进一步的 JAI 创建来生成缩放图像?

ParameterBlock paramBlock = new ParameterBlock();
paramBlock.addSource(((RenderedOp) pi).createInstance()); // Updated this
paramBlock.add(0.5f); // x Scale (Change these two Scale values!)
paramBlock.add(0.5f); // y Scale
paramBlock.add(0.0f); // x Translate
paramBlock.add(0.0f); // y Translate
paramBlock.add(new InterpolationBilinear()); // I think this Interpolation should work...)
RenderedOp resized = JAI.create("scale", paramBlock, null);

恐怕上面的代码完全未经测试作为标准,但应该让你开始!

一旦你得到了调整大小的 RenderedOp,你应该能够安全地将其转换为 BufferedImage(如果需要)。

于 2009-11-01T19:34:49.327 回答