在Java中向JPEG图像写入dpi元数据

15

我正在尝试在Java中以编程方式设置jpeg图像的dpi元数据。图像的来源是扫描仪,因此我从TWAIN获取水平/垂直分辨率以及图像原始数据。我想保存此信息以获得更好的打印结果。

这是我到目前为止拥有的代码。它将原始图像(byteArray)保存到JPEG文件中,但它忽略了我通过IIOMetadata指定的X/Ydensity信息。您有什么建议吗?

欢迎任何其他解决方案(第三方库等)。

import java.awt.image.BufferedImage;
import java.awt.image.WritableRaster;
import java.io.File;

import javax.imageio.IIOImage;
import javax.imageio.ImageIO;
import javax.imageio.ImageTypeSpecifier;
import javax.imageio.metadata.IIOMetadata;
import javax.imageio.plugins.jpeg.JPEGImageWriteParam;
import javax.imageio.stream.ImageOutputStream    

import org.w3c.dom.Element;    
import com.sun.imageio.plugins.jpeg.JPEGImageWriter;

public boolean saveJpeg(int[] byteArray, int width, int height, int dpi, String file)
{
    BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);

    WritableRaster wr = bufferedImage.getRaster();
    wr.setPixels(0, 0, width, height, byteArray);

    try
    {           
        // Image writer 
        JPEGImageWriter imageWriter = (JPEGImageWriter) ImageIO.getImageWritersBySuffix("jpeg").next();
        ImageOutputStream ios = ImageIO.createImageOutputStream(new File(file));
        imageWriter.setOutput(ios);

        // Compression
        JPEGImageWriteParam jpegParams = (JPEGImageWriteParam) imageWriter.getDefaultWriteParam();
        jpegParams.setCompressionMode(JPEGImageWriteParam.MODE_EXPLICIT);
        jpegParams.setCompressionQuality(0.85f);

        // Metadata (dpi)
        IIOMetadata data = imageWriter.getDefaultImageMetadata(new ImageTypeSpecifier(bufferedImage), jpegParams);
        Element tree = (Element)data.getAsTree("javax_imageio_jpeg_image_1.0");
        Element jfif = (Element)tree.getElementsByTagName("app0JFIF").item(0);
        jfif.setAttribute("Xdensity", Integer.toString(dpi));
        jfif.setAttribute("Ydensity", Integer.toString(dpi));
        jfif.setAttribute("resUnits", "1"); // density is dots per inch         

        // Write and clean up
        imageWriter.write(data, new IIOImage(bufferedImage, null, null), jpegParams);
        ios.close();
        imageWriter.dispose();
    }
    catch (Exception e)
    {
       return false;
    }

    return true;
}

谢谢!

2个回答

16

这里没有考虑到的一些问题:

  1. 树不直接映射到IOMetaData。要应用来自树的数据,请在设置密度和光栅参数后添加以下调用:

    data.setFromTree("javax_imageio_jpeg_image_1.0", tree);
    
  2. 不要在写入调用中将元数据作为第一个参数。请参见JPEGImageWriter#write(IIOMetaData, IIOImage, ImageWriteParam)。如果streamMetaData不为空,则会生成警告(WARNING_STREAM_METADATA_IGNORED)。

  3. 将元数据设置为IOImageIOMetadata。这些元数据由JPEGImageWriter使用。正确的写入调用如下:

    imageWriter.write(null, new IIOImage(F_scaledImg, null, data), jpegParams);
    

2
F_scaledImg是在哪里设置的?它和bufferedImage是一样的吗?如果这是一个愚蠢的问题,我很抱歉... - Paul Morrison
2
是的,实际上它是缓冲图像。Christoph可能复制粘贴了代码片段并忘记编辑它。 - dasp

2
我看起来这可能是一个错误。
我从几个谷歌搜索中找到了这篇文章(链接)
显然还有很多其他指向错误的文章。
上面的文章谈到使用JMagick作为第三方解决方案。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接