Манипуляции с изображением без удаления его данных EXIF

При использовании imageIO у меня обычно возникает проблема с преобразованием файла изображения, и после его перезаписи он теряет все свои данные EXIF. Есть ли способ сохранить его без предварительного извлечения, кэширования и последующего сброса?


person Preslav Rachev    schedule 23.01.2012    source источник
comment
Сохранить его в другом месте, а затем перезаписать новое изображение старой метамета exif? screaming-penguin.com/node/7485   -  person Sergey Benner    schedule 23.01.2012
comment
именно этого я и хочу избежать   -  person Preslav Rachev    schedule 23.01.2012
comment
в чем проблема скопировать мета? вот еще один пример nucleussystems.com/blog/java-copy-exif-data   -  person Sergey Benner    schedule 23.01.2012
comment
Библиотеки, которые я нашел до сих пор, либо читают метаданные, либо читают изображение. Не оба. Вам придется прочитать его дважды. Если он читается из потока, это означает, что вам нужно сохранить его в byte[]. Что может потребовать слишком много дополнительной памяти.   -  person Ed Randall    schedule 16.05.2014


Ответы (2)


ImageIO сама имеет эту функцию, но вместо ImageIO.read вам нужно будет использовать ImageReader:

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.

Поскольку это старая тема, я думаю, что этот ответ слишком запоздал, но может помочь другим, поскольку эта тема все еще доступна для поиска в Google.

person Rigeborod    schedule 17.04.2016
comment
Это выглядит значительно более эффективным с точки зрения памяти, чем мое решение, я думаю, что одна копия изображения все еще находится в памяти, но на самом деле этого нельзя избежать. - person Ed Randall; 20.05.2016
comment
Только что наткнулся на это и попытался использовать его ... не уверен, что последняя строка, ImageIO.write(...), должна быть там, поскольку кажется, что изображение переписывается без метаданных. Просто добавлю, вдруг кто наткнется. - person cfnz; 26.02.2020
comment
Что ж, вы правы и не правы одновременно. Он действительно запишет изображение без метаданных... но в конце потока. Эффективно удваивает размер файла, но сохраняет правильное изображение с метаданными... - person Rigeborod; 17.06.2020

Вот мое решение, использующее комбинацию 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;
    }

}
person Ed Randall    schedule 19.05.2014
comment
Большое спасибо. Это сработало хорошо. Единственное, что, по-видимому, IImageMetadata называется ImageMetadata в текущем репо для Apache Commons Imaging. - person m.hashemian; 30.08.2016
comment
Также проверьте другое решение от @Rigeborod, которое выглядит немного более эффективным. - person Ed Randall; 06.09.2017