java - 在 Java 中使用 WritableRaster 和 DataBufferByte 将图像转换为 byte[] 时出错。

标签 java image bufferedimage javax.imageio

我正在尝试使用代码将图像转换为 byte[]

public static byte[] extractBytes(String ImageName) throws IOException {
    // open image
    File imgPath = new File(ImageName);
    BufferedImage bufferedImage = ImageIO.read(imgPath);

    // get DataBufferBytes from Raster
    WritableRaster raster = bufferedImage.getRaster();
    DataBufferByte data = (DataBufferByte) raster.getDataBuffer();

    return (data.getData());
}

当我使用代码测试它时

public static void main(String[] args) throws IOException {

    String filepath = "image_old.jpg";
    byte[] data = extractBytes(filepath);
    System.out.println(data.length);
    BufferedImage img = ImageIO.read(new ByteArrayInputStream(data));
    File outputfile = new File("image_new.jpg");
    ImageIO.write(img, "jpeg", outputfile);
}

我收到 data.length = 4665600 并收到错误

Exception in thread "main" java.lang.IllegalArgumentException: image == null!
    at javax.imageio.ImageTypeSpecifier.createFromRenderedImage(ImageTypeSpecifier.java:925)
    at javax.imageio.ImageIO.getWriter(ImageIO.java:1591)
    at javax.imageio.ImageIO.write(ImageIO.java:1520)
    at com.medianet.hello.HbaseUtil.main(HbaseUtil.java:138)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

但是当我将 extractBytes 代码更改为

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

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

        return baos.toByteArray();
    }

我得到 data.length = 120905 并获得成功(image.jpg 在所需位置创建)

最佳答案

问题是,extractBytes 的第一个版本读取图像,并且仅将图像的像素作为字节数组返回(假设它使用DataBufferByte)。这些字节不是文件格式,如果没有额外的信息(例如宽度、高度、颜色空间等),这些字节是无用的。ImageIO无法读回这些字节,因此,返回 null(并分配给 img,稍后导致来自 ImageIO.write(...)IllegalArgumentException)。

第二个版本对图像进行解码,然后再次以 JPEG 格式对其进行编码。这是 ImageIO 能够读取的格式,并且您将获得您期望的图像(分配给 img)。

但是,您的代码似乎只是一种非常非常消耗 CPU 的图像复制方式(您解码图像,然后编码,然后再次解码,最后编码)...对于 JPEG 文件,此解码/编码周期将也会降低图像质量。除非您打算将图像数据用于任何用途,并且只想将图像从一个位置复制到另一个位置,否则不要使用 ImageIOBufferedImage。这些类型用于图像处理。

这是 main 方法的修改版本:

public static void main(String[] args) throws IOException {
    byte[] buffer = new byte[1024];

    File inFile = new File("image_old.jpg");
    File outFile = new File("image_new.jpg");

    InputStream in = new FileInputStream(inFile);
    try {
        OutputStream out = new FileOutputStream(outFile);

        try {
            int len;
            while ((len = in.read(buffer)) > 0) {
                out.write(buffer, 0, len);
            }
        }
        finally {
            out.close();
        }
    }
    finally {
        in.close();
    }
}

(在 Java 7 中使用 try-with-resources 或在 Java 8 中使用 NIO2 Files.copy 可以更好/更优雅地编写此代码,但我将其留给您。:-) )

关于java - 在 Java 中使用 WritableRaster 和 DataBufferByte 将图像转换为 byte[] 时出错。,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29119531/

相关文章:

java - 使用 JUnit 开始 Java 测试

java - 为什么一个地方的原始类型会导致其他地方的通用调用站点被视为原始类型?

图像 css 最大宽度没有扭曲

java - 使 BufferedImage 中的黑色像素变亮

java - 如何向外部 Rest Api 发出 Http Post 请求?

css - 为什么不对作为页面内容的较大图像进行 sprite?

android - android中的灰度位图

java - 如何在 java 中使用 apache.commons.imaging 保留 TIFF 图像的透明度

java - JAVA中像素改变后的图像实现

Java - 比较两种 O(n) 算法的效率