如何在图像中设置DPI信息?

31

我有一个应用程序,我想要导出高分辨率(或者说是高像素密度?)的图片以供打印 - 例如,我希望图像的打印分辨率为每英寸250个点(DPI),而不是默认的72 DPI。

我正在使用一个BufferedImage对象和一个Graphics2D对象来绘制图像,然后使用ImageIO.write()来保存图像。

你知道如何设置DPI吗?

3个回答

33

Kurt的答案指明了方向,但我花了相当长的时间才让它运行起来,所以这里是设置保存PNG时DPI的代码。还有很多工作要做才能获得合适的写入器等...

 private BufferedImage gridImage;
 ...

 private void saveGridImage(File output) throws IOException {
    output.delete();

    final String formatName = "png";

    for (Iterator<ImageWriter> iw = ImageIO.getImageWritersByFormatName(formatName); iw.hasNext();) {
       ImageWriter writer = iw.next();
       ImageWriteParam writeParam = writer.getDefaultWriteParam();
       ImageTypeSpecifier typeSpecifier = ImageTypeSpecifier.createFromBufferedImageType(BufferedImage.TYPE_INT_RGB);
       IIOMetadata metadata = writer.getDefaultImageMetadata(typeSpecifier, writeParam);
       if (metadata.isReadOnly() || !metadata.isStandardMetadataFormatSupported()) {
          continue;
       }

       setDPI(metadata);

       final ImageOutputStream stream = ImageIO.createImageOutputStream(output);
       try {
          writer.setOutput(stream);
          writer.write(metadata, new IIOImage(gridImage, null, metadata), writeParam);
       } finally {
          stream.close();
       }
       break;
    }
 }

 private void setDPI(IIOMetadata metadata) throws IIOInvalidTreeException {

    // for PMG, it's dots per millimeter
    double dotsPerMilli = 1.0 * DPI / 10 / INCH_2_CM;

    IIOMetadataNode horiz = new IIOMetadataNode("HorizontalPixelSize");
    horiz.setAttribute("value", Double.toString(dotsPerMilli));

    IIOMetadataNode vert = new IIOMetadataNode("VerticalPixelSize");
    vert.setAttribute("value", Double.toString(dotsPerMilli));

    IIOMetadataNode dim = new IIOMetadataNode("Dimension");
    dim.appendChild(horiz);
    dim.appendChild(vert);

    IIOMetadataNode root = new IIOMetadataNode("javax_imageio_1.0");
    root.appendChild(dim);

    metadata.mergeTree("javax_imageio_1.0", root);
 }

@Zeveso 我只是偶然从一个保存网格图像的工作应用程序中复制/粘贴了这段代码,因此有了BufferedImage的名称和方法。该代码适用于每个BufferedImage。 - Peter Kofler
1
@PeterKofler 我正在使用上述代码,没有做任何更改。上述代码按预期创建PNG文件(使用给定的dpi),但当我将上述代码用于JPEG文件时,它失败了,我需要为JPEG设置任何参数吗?还是这个代码只适用于PNG? - Mihir
@Mihir,我不知道JPG会有什么问题。你能更具体地说明它是如何失败的吗? - Peter Kofler
1
当我使用上述代码将JPEG文件保存为400 DPI时,通过设置formatName =“jpg”,它会成功保存文件。但是,当我在Irfanview、Photoshop和Inkspace中打开这个已保存的文件时,都显示文件的DPI为72 DPI。如果需要,我可以在SO上发布一个相关的截图并提出单独的问题。 - Mihir
@Mihir,你有进展了吗? - planty182
显示剩余3条评论

4

设置TIFF DPI

如果您想为TIFF设置dpi,请按以下步骤操作:

private static IIOMetadata createMetadata(ImageWriter writer, ImageWriteParam writerParams, int resolution) throws
                                                                                                            IIOInvalidTreeException
{
    // Get default metadata from writer
    ImageTypeSpecifier type = ImageTypeSpecifier.createFromBufferedImageType(BufferedImage.TYPE_BYTE_GRAY);
    IIOMetadata meta = writer.getDefaultImageMetadata(type, writerParams);

    // Convert default metadata to TIFF metadata
    TIFFDirectory dir = TIFFDirectory.createFromMetadata(meta);

    // Get {X,Y} resolution tags
    BaselineTIFFTagSet base = BaselineTIFFTagSet.getInstance();
    TIFFTag tagXRes = base.getTag(BaselineTIFFTagSet.TAG_X_RESOLUTION);
    TIFFTag tagYRes = base.getTag(BaselineTIFFTagSet.TAG_Y_RESOLUTION);

    // Create {X,Y} resolution fields
    TIFFField fieldXRes = new TIFFField(tagXRes, TIFFTag.TIFF_RATIONAL, 1, new long[][] { { resolution, 1 } });
    TIFFField fieldYRes = new TIFFField(tagYRes, TIFFTag.TIFF_RATIONAL, 1, new long[][] { { resolution, 1 } });

    // Add {X,Y} resolution fields to TIFFDirectory
    dir.addTIFFField(fieldXRes);
    dir.addTIFFField(fieldYRes);

    // Add unit field to TIFFDirectory (change to RESOLUTION_UNIT_CENTIMETER if necessary)
    dir.addTIFFField(new TIFFField(base.getTag(BaselineTIFFTagSet.TAG_RESOLUTION_UNIT), BaselineTIFFTagSet.RESOLUTION_UNIT_INCH));

    // Return TIFF metadata so it can be picked up by the IIOImage
    return dir.getAsMetadata();
}

同样的方法,您可以设置任何TIFF标签。
阅读更多请参考

我认为代码不完整,因为它没有设置分辨率单位,默认值为 RESOLUTION_UNIT_NONE。在这种情况下,阅读器/客户端应将比率设置为 1/1,请参见 TIFFImageWriter:1182。修复方法: dir.addTIFFField(new TIFFField(base.getTag(BaselineTIFFTagSet.TAG_RESOLUTION_UNIT), BaselineTIFFTagSet.RESOLUTION_UNIT_INCH))。免责声明:我不是TIFF专家。 - dma_k
@dma_k 如果你的建议可行,请随意编辑答案。我也不是TIFF专家) - Sergei Bubenshchikov
作为额外的参考,我提供了一个链接TiffImage:146,在这里可以看到,如果分辨率单位为“无”,则X/Y DPI设置为零,因此iTextPDF库将图像解释为未设置DPI。已知哪些客户端能正常工作? - dma_k

3

我在我的项目中使用这段代码来处理TIFF文件,它运行良好。

import java.awt.image.BufferedImage;
import java.awt.image.RenderedImage;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.List;
import javax.imageio.ImageIO;
import javax.media.jai.NullOpImage;
import javax.media.jai.OpImage;
import javax.media.jai.PlanarImage;
import com.sun.media.jai.codec.FileSeekableStream;
import com.sun.media.jai.codec.ImageCodec;
import com.sun.media.jai.codec.ImageDecoder;
import com.sun.media.jai.codec.ImageEncoder;
import com.sun.media.jai.codec.SeekableStream;
import com.sun.media.jai.codec.TIFFEncodeParam;
import com.sun.media.jai.codec.TIFFField;
class SetDDPI
 {
static void tiff_Maker(List<BufferedImage> output, String result) throws   IOException
{
    TIFFEncodeParam params = new TIFFEncodeParam();
    OutputStream out = new FileOutputStream(result);
    List<BufferedImage> imageList = new ArrayList<BufferedImage>();
    for (int i = 1; i < output.size(); i++)
    {
        imageList.add(output.get(i));
    }
    params.setWriteTiled(true);
    params.setCompression(TIFFEncodeParam.COMPRESSION_GROUP4);
    params.setExtraImages(imageList.iterator());
    TIFFField[] extras = new TIFFField[2];
    extras[0] = new TIFFField(282, TIFFField.TIFF_RATIONAL, 1, (Object) new long[][] { { (long) 300, (long) 1 },
            { (long) 0, (long) 0 } });
    extras[1] = new TIFFField(283, TIFFField.TIFF_RATIONAL, 1, (Object) new long[][] { { (long) 300, (long) 1 },
            { (long) 0, (long) 0 } });
    params.setExtraFields(extras);
    ImageEncoder encoder = ImageCodec.createImageEncoder("tiff", out, params);
    encoder.encode(output.get(0));
    out.close();
}
static List<BufferedImage> tiff_Extractor(File tiff) throws IOException
{
    List<BufferedImage> images = new ArrayList<BufferedImage>();
    SeekableStream ss = new FileSeekableStream(tiff);
    ImageDecoder decoder = ImageCodec.createImageDecoder("tiff", ss, null);
    int numPages = decoder.getNumPages();
    for (int j = 0; j < numPages; j++)
    {
        PlanarImage op = new NullOpImage(decoder.decodeAsRenderedImage(j), null, null, OpImage.OP_IO_BOUND);
        images.add(op.getAsBufferedImage());

    }
    return images;
}
}

这是设置Tiff图像为300 DPI的方法。您可以根据需要进行更改。

extras[0] = new TIFFField(282, TIFFField.TIFF_RATIONAL, 1, (Object) new     
long[][] { { (long) 300, (long) 1 },{ (long) 0, (long) 0 } });

extras[1] = new TIFFField(283, TIFFField.TIFF_RATIONAL, 1, (Object) new     
long[][] { { (long) 300, (long) 1 },{ (long) 0, (long) 0 } });

老兄,你不知道这提供了多少帮助……非常感谢你分享这个!! - Dan Ortega
这对于设置 DPI 很好用,但我如何设置分辨率? - user2179092
@rj27 我尝试使用tiff_maker中的上述代码为我拥有的bufferedImage设置dpi,然后我尝试将其保存到本地文件系统writer.write(null, new IIOImage(image, null, null), null);。之后我尝试获取图像,但是我得到了一个非常大的灰色屏幕。我是否漏掉了什么? - xyzzz

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