Warning: file_get_contents(/data/phpspider/zhask/data//catemap/7/image/5.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java 使用BuffereImage和ImageIO将图像转换为字节[]后,图像大小减小_Java_Image_Bufferedimage_Javax.imageio - Fatal编程技术网

Java 使用BuffereImage和ImageIO将图像转换为字节[]后,图像大小减小

Java 使用BuffereImage和ImageIO将图像转换为字节[]后,图像大小减小,java,image,bufferedimage,javax.imageio,Java,Image,Bufferedimage,Javax.imageio,我正在使用以下代码将图像转换为字节[] public static byte[] extractBytes (String ImageName) throws IOException { ByteArrayOutputStream baos=new ByteArrayOutputStream(); BufferedImage img=ImageIO.read(new File(ImageName)); ImageIO.write(img, "jpg", b

我正在使用以下代码将图像转换为字节[]

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
将图像转换为字节[]后,其大小将减小约1/4,并且当我将字节[]转换回图像时,其大小也会发生变化。但输出图像已成功地在所需位置创建。在放大500-600%后,我可以看到原始图像和新图像的质量略有不同。放大后,新图像有点模糊

这是我正在进行测试的图像


请解释大小更改的原因,我也想知道在此之后获得相同大小的任何方法。

您拥有的图像以最大质量设置100%或1.0(图像IO术语)进行压缩。JPEG压缩在如此高的设置下不是非常有效,因此比通常的要大一些。使用ImageIO.write…、JPEG。。。将使用默认的质量设置。此默认值为0.75。但此值的确切含义取决于编码器,且不精确,因此质量较低,导致文件大小较小

原始图像和重新压缩图像之间的文件大小显著减小的另一个可能原因是元数据的删除。当使用ImageIO.readfile读取时,实际上是在剥离JPEG文件中的所有元数据,如XMP、Exif或ICC配置文件。在极端情况下,是的,我这里主要讲的是Photoshop;-此元数据可能比图像数据本身占用更多空间,即可能占用兆字节的元数据。但是,您的文件并非如此

从字节[]到最终输出文件的第二次重新压缩中可以看到,输出比输入稍微小一些。这是因为未指定质量设置,所以在这两种情况下仍然使用默认值也是一样的,在这一步中,任何元数据也将丢失,因此不会添加到文件大小。微小差异可能是由于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();
}

您拥有的图像是以最大质量设置100%或ImageIO术语中的1.0进行压缩的。JPEG压缩在如此高的设置下不是非常有效,因此比通常的要大一些。使用ImageIO.write…、JPEG。。。将使用默认的质量设置。此默认值为0.75。但此值的确切含义取决于编码器,且不精确,因此质量较低,导致文件大小较小

原始图像和重新压缩图像之间的文件大小显著减小的另一个可能原因是元数据的删除。当使用ImageIO.readfile读取时,实际上是在剥离JPEG文件中的所有元数据,如XMP、Exif或ICC配置文件。在极端情况下,是的,我这里主要讲的是Photoshop;-此元数据可能比图像数据本身占用更多空间,即可能占用兆字节的元数据。但是,您的文件并非如此

从字节[]到最终输出文件的第二次重新压缩中可以看到,输出比输入稍微小一些。这是因为未指定质量设置,所以在这两种情况下仍然使用默认值也是一样的,在这一步中,任何元数据也将丢失,因此不会添加到文件大小。微小差异可能是由于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();
}

调用ImageIO.writeimg、jpeg、outputfile时;李彦宏 brary使用自己的压缩参数编写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在调用ImageIO.writeimg、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在这里给出的答案改编的

这就是压缩的工作原理。如果将同一内容压缩两次,可能无法获得完全相同的压缩文件两次。此外,使用JPEG压缩图像时,JPEG文件的质量低于原始图像。@immibis,但为什么即使我没有添加任何压缩代码,它也会被压缩。你能详细解释一下这里发生了什么吗。我很困惑。@immibis,是的,质量也在下降。在对两幅图像进行600%缩放后,我意识到:具有读写功能的PImageIO为多种文件类型提供了插件机制。你可以使用ImageWriter for JPEG并设置选项,比如高质量。这就是压缩的工作原理。如果将同一内容压缩两次,可能无法获得完全相同的压缩文件两次。此外,使用JPEG压缩图像时,JPEG文件的质量低于原始图像。@immibis,但为什么即使我没有添加任何压缩代码,它也会被压缩。你能详细解释一下这里发生了什么吗。我很困惑。@immibis,是的,质量也在下降。在对两幅图像进行600%缩放后,我意识到:具有读写功能的PImageIO为多种文件类型提供了插件机制。你可以使用ImageWriter for JPEG并使用选项集进行编写,比如高质量。嘿,我已经编辑了我的答案并添加了我正在进行测试的图像。是的,这就是我现在使用的方法。谢谢,我来晚了。。但另一个信息。。当使用ImageIO.write时,我得到了一个倒转的图像文件和一个2.3 Mb->23 Mb的大尺寸图像。。我知道我做错了什么。。但是找不到@BhanuchanderUdhayakumar很可能您的原始文件包含Exif信息,指定图像方向。大多数软件都会根据这一点为您旋转图像。ImageIO没有。因此,当您在没有Exif的情况下再次写入图像时,将不再应用方向,图像将出现倒置或仅旋转90度。@haraldK感谢您的回复。我还得到了图像文件的大小增加。大约10倍。这是如何在字节数组中发生的。嘿,我已经编辑了我的答案并添加了我正在测试的图像。是的,这就是我现在使用的方法。谢谢,我来晚了。。但另一个信息。。当使用ImageIO.write时,我得到了一个倒转的图像文件和一个2.3 Mb->23 Mb的大尺寸图像。。我知道我做错了什么。。但是找不到@BhanuchanderUdhayakumar很可能您的原始文件包含Exif信息,指定图像方向。大多数软件都会根据这一点为您旋转图像。ImageIO没有。因此,当您在没有Exif的情况下再次写入图像时,将不再应用方向,图像将出现倒置或仅旋转90度。@haraldK感谢您的回复。我还得到了图像文件的大小增加。大约10倍。字节数组中是如何发生这种情况的..嘿,谢谢你的精彩回复,但是如果我使用的是jpegParams,我仍然感到困惑。setCompressionQuality1f,在这种情况下,我的图像大小仍然在变大。为什么我的新图像大小与原始图像大小不同?@harshharma,因为您仍在对JPEG进行解码/重新编码。相反,如果你想要一份准确的副本,按照我在回答中的建议去做。只需复制数据,就像任何其他二进制文件一样。嘿,谢谢你的精彩回复,但如果我使用的是jpegParams,我仍然感到困惑。setCompressionQuality1f,我的图像大小仍然在变。在这种情况下,图像大小正在变大。为什么我的新图像大小与原始图像大小不同?@harshharma,因为您仍在对JPEG进行解码/重新编码。相反,如果你想要一份准确的副本,按照我在回答中的建议去做。只需复制数据,就像任何其他二进制文件一样。