【问题标题】:Manipulate an image without deleting its EXIF data处理图像而不删除其 EXIF 数据
【发布时间】:2012-02-16 20:17:09
【问题描述】:

使用 imageIO,我通常会遇到转换图像文件的问题,并且在覆盖它之后,它会丢失所有的 EXIF 数据。有什么方法可以在不先提取、缓存然后重置的情况下保存它?

【问题讨论】:

  • 将其存储在其他地方,然后用旧的 exif 元数据覆盖新图像? screaming-penguin.com/node/7485
  • 这正是我想要避免的
  • 复制元数据有什么问题?这是另一个例子nucleussystems.com/blog/java-copy-exif-data
  • 目前我发现的库要么读取元数据,要么读取图像。不是都。你必须读两遍。如果它是从流中读取的,这意味着您需要将其保存在一个字节 [] 中。这可能需要太多额外的内存。

标签: java bitmap bufferedimage image-resizing javax.imageio


【解决方案1】:

ImageIO 本身确实具有此功能,但您需要使用 ImageReader 而不是 ImageIO.read

ImageReader reader = ImageIO.getImageReadersBySuffix("jpg").next();

(您可能还想检查这样的阅读器是否存在)。 然后你需要设置输入:

reader.setInput(ImageIO.createImageInputStream(your_imput_stream));

现在您可以保存元数据了:

IIOMetadata metadata = reader.getImageMetadata(0); 
                            // As far as I understand you should provide 
                            // index as tiff images could have multiple pages

然后读取图片:

BufferedImage bi = reader.read(0);

当你想保存新图像时,你应该使用 ImageWriter:

// I'm writing to byte array in memory, but you may use any other stream
ByteArrayOutputStream os = new ByteArrayOutputStream(255);
ImageOutputStream ios = ImageIO.createImageOutputStream(os);

Iterator<ImageWriter> iter = ImageIO.getImageWritersByFormatName("jpeg");
ImageWriter writer = iter.next();
writer.setOutput(ios);

//You may want also to alter jpeg quality
ImageWriteParam iwParam = writer.getDefaultWriteParam();
iwParam.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
iwParam.setCompressionQuality(.95f);

//Note: we're using metadata we've already saved.
writer.write(null, new IIOImage(bi, null, metadata), iwParam);
writer.dispose();

//ImageIO.write(bi, "jpg", ios); <- This was initially in the code but actually it was only adding image again at the end of the file.

由于它是老话题,我想这个答案有点太晚了,但可能会帮助其他人,因为这个话题仍然可以谷歌搜索。

【讨论】:

  • 这看起来比我的解决方案更节省内存,我猜图像的一个副本仍在内存中,但无法真正避免这种情况。
  • 刚刚遇到这个并尝试使用它...不确定最后一行 ImageIO.write(...) 是否应该在那里,因为它似乎在没有元数据的情况下重写了图像。只是添加以防有人偶然发现它。
  • 嗯,你是对的,也是错的。它确实会在没有元数据的情况下写入图像......但在流的末尾。有效地将文件大小加倍,但保留正确的图像和元数据...
【解决方案2】:

这是我结合使用 ImageIO、Imgscalr 和 Apache commons-imaging 的解决方案。遗憾的是,没有一个库可以将读取图像及其元数据结合起来,这可能会导致内存使用过多;欢迎改进。

import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import javax.imageio.ImageIO;
import org.apache.commons.imaging.ImageReadException;
import org.apache.commons.imaging.ImageWriteException;
import org.apache.commons.imaging.Imaging;
import org.apache.commons.imaging.common.IImageMetadata;
import org.apache.commons.imaging.formats.jpeg.JpegImageMetadata;
import org.apache.commons.imaging.formats.jpeg.exif.ExifRewriter;
import org.apache.commons.imaging.formats.tiff.TiffImageMetadata;
import org.apache.commons.io.IOUtils;
import org.imgscalr.Scalr;


public class ImageData {

    private byte[] imageData;


    public ImageData(InputStream instream) throws IOException {
        imageData  = IOUtils.toByteArray(instream);
        instream.close();
    }


    public synchronized void resize(int maxDimension) throws IOException, ImageReadException, ImageWriteException {
        // Resize the image if necessary
        BufferedImage image = readImage(imageData);
        if (image.getWidth() > maxDimension || image.getHeight() > maxDimension) {

            // Save existing metadata, if any
            TiffImageMetadata metadata = readExifMetadata(imageData);
            imageData = null; // allow immediate GC

            // resize 
            image = Scalr.resize(image, maxDimension);

            // rewrite resized image as byte[]
            byte[] resizedData = writeJPEG(image);
            image = null; // allow immediate GC 

            // Re-code resizedData + metadata to imageData 
            if (metadata != null) {
                this.imageData = writeExifMetadata(metadata, resizedData);
            } else {
                this.imageData = resizedData;
            }
        }
    }

    private TiffImageMetadata readExifMetadata(byte[] jpegData) throws ImageReadException, IOException {
        IImageMetadata imageMetadata = Imaging.getMetadata(jpegData);
        if (imageMetadata == null) {
            return null;
        }
        JpegImageMetadata jpegMetadata = (JpegImageMetadata)imageMetadata;
        TiffImageMetadata exif = jpegMetadata.getExif();
        if (exif == null) {
            return null;
        }
        return exif;
    }


    private byte[] writeExifMetadata(TiffImageMetadata metadata, byte[] jpegData) 
                                throws ImageReadException, ImageWriteException, IOException {
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        new ExifRewriter().updateExifMetadataLossless(jpegData, out, metadata.getOutputSet());
        out.close();
        return out.toByteArray();
    }


    private BufferedImage readImage(byte[] data) throws IOException {
        return ImageIO.read(new ByteArrayInputStream(data));
    }

    private byte[] writeJPEG(BufferedImage image) throws IOException {
        ByteArrayOutputStream jpegOut = new ByteArrayOutputStream();
        ImageIO.write(image, "JPEG", jpegOut);
        jpegOut.close();
        return jpegOut.toByteArray();
    }

    public synchronized void writeJPEG(OutputStream outstream) throws IOException {
        IOUtils.write(imageData,  outstream);

    }

    public synchronized byte[] getJPEGData() {
        return imageData;
    }

}

【讨论】:

  • 非常感谢。它工作得很好。唯一的问题是 IImageMetadata 在当前的 Apache Commons Imaging 存储库中被命名为 ImageMetadata
  • 还要检查@Rigeborod 的其他解决方案,它看起来更有效率
猜你喜欢
  • 2013-11-17
  • 1970-01-01
  • 1970-01-01
  • 2015-02-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-11-16
相关资源
最近更新 更多