使用BufferedImage和ImageIO将图像大小转换为byte []后图像大小减小

时间:2015-03-26 08:11:53

标签: java image bufferedimage javax.imageio

我正在使用以下代码将Image转换为byte []。

public static byte[] extractBytes (String ImageName) throws IOException {

   ByteArrayOutputStream baos=new ByteArrayOutputStream();
        BufferedImage img=ImageIO.read(new File(ImageName));
        ImageIO.write(img, "jpg", baos);
        return baos.toByteArray();
    }

现在我正在测试我的代码:

public static void main(String[] args) throws IOException {
    String filepath = "image_old.jpg";
    File outp=new File(filepath);
    System.out.println("Size of original image="+outp.length());
    byte[] data = extractBytes(filepath);
    System.out.println("size of byte[] data="+data.length);
    BufferedImage img = ImageIO.read(new ByteArrayInputStream(data));
    //converting the byte[] array into image again
    File outputfile = new File("image_new.jpg");
    ImageIO.write(img, "jpeg", outputfile);
    System.out.println("size of converted image="+outputfile.length());
}

我得到了非常奇怪的结果:

Size of original image=78620 
size of byte[] data=20280
size of converted image=20244

将图像转换为byte []后,其大小减少了大约1/4,当我将byte []转换回图像时,其大小也会改变。但输出图像已成功在所需位置创建。放大500-600%后,我可以看到原始图像和新图像的质量略有差异。放大后新图像很少模糊。

以下是我正在进行测试http://pbrd.co/1BrOVbf

的图片

请解释这个尺寸变化的原因,并且我想知道在此之后获得相同尺寸的任何方法。

2 个答案:

答案 0 :(得分:2)

您拥有的图像采用最高质量设置进行压缩(" 100%"或{IMO}中的1.0。 JPEG压缩在如此高的设置下并不是非常有效,因此比平时要大得多。使用ImageIO.write(..., "JPEG", ...)时,将使用默认质量设置。这个默认值是0.75(这个值的确切含义虽然与解码器有关,但并不是精确的科学),因而质量较低,导致文件较小。

(原始图像和重新压缩图像之间文件大小显着减少的另一个可能原因是元数据的删除。使用ImageIO.read(file)阅读时,您实际上正在删除任何元数据。 JPEG文件,如XMP,Exif或ICC配置文件。在极端情况下(是的,我在这里主要讨论Photoshop ;-))这个元数据可以占用比图像数据本身更多的空间(即兆字节的元数据)数据是可能的)。但是,这不是您的文件的情况。)

从第二次重新压缩(从byte[]到最终输出文件)可以看出,输出略小于输入。这是因为质量设置(未指定,因此仍然使用默认值)在两种情况下都是相同的(此外,在此步骤中也会丢失任何元数据,因此不会添加到文件大小)。微小差异可能是由于JPEG解压缩/重新压缩中的一些小损失(舍入误差等)。

虽然稍微违反直觉,但是在重新压缩JPEG时,最少的数据丢失(就原始图像的变化而言,不是文件大小而言)总是通过使用相同的质量设置进行重新压缩来实现(使用完全相同的表应该是无损的,但可能仍会出现小的舍入错误)。提高质量设置会使文件输出更大,但质量实际上会降低。

100%确保不丢失任何数据或图像质量的唯一方法是首先不对图像进行解码/编码,而是逐字节地复制文件,例如:

File in = ...;
File out = ...;

InputStream input = new FileInputStream(in);
try {
    OutputStream output = new FileOutputStream(out);
    try {
        copy(input, output);
    }
    finally {
        output.close();
    }
}
finally {
    input.close();
}

复制方法:

public void copy(final InputStream in, final OutputStream out) {
    byte[] buffer = new byte[1024]; 
    int count;

    while ((count = in.read(buffer)) != -1) {
        out.write(buffer, 0, count);
    }

    // Flush out stream, to write any remaining buffered data
    out.flush();
}

答案 1 :(得分:1)

当您调用ImageIO.write(img," jpeg",outputfile)时; ImageIO库使用自己的压缩参数写入jpeg图像。输出图像似乎比输入图像压缩得更多。您可以通过更改下面调用jpegParams.setCompressionQuality中的参数来调整压缩级别。生成的文件可能比原始文件更大或更小,具体取决于每个文件中的相对压缩级别。

public static ImageWriter getImageWriter() throws IOException {
    IIORegistry registry = IIORegistry.getDefaultInstance();
    Iterator<ImageWriterSpi> services = registry.getServiceProviders(ImageWriterSpi.class, (provider) -> {
        if (provider instanceof ImageWriterSpi) {
            return Arrays.stream(((ImageWriterSpi) provider).getFormatNames()).anyMatch(formatName -> formatName.equalsIgnoreCase("JPEG"));
        }
        return false;
    }, true);
    ImageWriterSpi writerSpi = services.next();
    ImageWriter writer = writerSpi.createWriterInstance();
    return writer;
}

public static void main(String[] args) throws IOException {
    String filepath = "old.jpg";
    File outp = new File(filepath);
    System.out.println("Size of original image=" + outp.length());
    byte[] data = extractBytes(filepath);
    System.out.println("size of byte[] data=" + data.length);
    BufferedImage img = ImageIO.read(new ByteArrayInputStream(data));
    File outputfile = new File("new.jpg");
    JPEGImageWriteParam jpegParams = new JPEGImageWriteParam(null);
    jpegParams.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
    jpegParams.setCompressionQuality(1f);
    ImageWriter writer = getImageWriter();
    outputfile.delete();
    try (final ImageOutputStream stream = createImageOutputStream(outputfile)) {
        writer.setOutput(stream);
        try {
            writer.write(null, new IIOImage(img, null, null), jpegParams);
        } finally {
            writer.dispose();
            stream.flush();
        }
    }
    System.out.println("size of converted image=" + outputfile.length());
}

此解决方案改编自JeanValjean在此处Setting jpg compression level with ImageIO in Java

给出的答案