java - 在 Java 中以较小的磁盘大小写入 PNG 文件

标签 java image image-processing png

我有一个BufferedImage:

BufferedImage bi = new BufferedImage(14400, 14400, BufferedImage.TYPE_INT_ARGB);

我已使用以下代码将此图像保存为 PNG 文件:

public static void saveGridImage(BufferedImage sourceImage, int DPI,
            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, DPI);

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

    public static void setDPI(IIOMetadata metadata, int DPI)
            throws IIOInvalidTreeException {

        double INCH_2_CM = 2.54;

        // for PNG, 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);
    }

当代码执行时,它会创建一个 400 DPI 和 168 MB 磁盘大小的 PNG 文件;这太过分了。

有什么方法或参数可以用来保存较小的 PNG 吗?

之前,我有一个 1.20 GB 的 TIFF 文件,当我使用 imagemagick 以 400 DPI 将其转换为 PNG 时,生成的文件大小仅为 700 KB。

所以,我想我可以将上面的文件保存得更小一些。

pngj 可以帮助我吗?因为我现在有一个可以在 pngj 库中读取的 png 文件。

最佳答案

14400x14400 ARGB8 图像的原始(未压缩)大小为 791MB。它将根据其性质(具有均匀或平滑的区域)和(不太重要的)PNG 压缩参数进行或多或少的压缩。

when i convert it using imagemagic to PNG using 400 DPI , the resulting file size is only 700 KB.

(我不明白你为什么说 DPI,这无关紧要,重要的是像素大小)你是说你得到的是 700KB 的 14400x14400 ARGB?这将代表 1/1000 的压缩,除非图像实际上是平坦的,否则很难相信。 您应该首先了解这里发生了什么。

无论如何,这是一个 PNGJ 的示例代码

/** writes a BufferedImage of type TYPE_INT_ARGB to PNG using PNGJ */
public static void writeARGB(BufferedImage bi, OutputStream os) {
    if(bi.getType() != BufferedImage.TYPE_INT_ARGB) 
       throw new PngjException("This method expects  BufferedImage.TYPE_INT_ARGB" );
    ImageInfo imi = new ImageInfo(bi.getWidth(), bi.getHeight(), 8, true);
    PngWriter pngw = new PngWriter(os, imi);
    pngw.setCompLevel(9);// maximum compression, not critical usually
    pngw.setFilterType(FilterType.FILTER_AGGRESSIVE); // see what you prefer here
    DataBufferInt db =((DataBufferInt) bi.getRaster().getDataBuffer());
    SinglePixelPackedSampleModel samplemodel =  (SinglePixelPackedSampleModel) bi.getSampleModel();
    if(db.getNumBanks()!=1) 
        throw new PngjException("This method expects one bank");
    ImageLine line = new ImageLine(imi);
    for (int row = 0; row < imi.rows; row++) {
        int elem=samplemodel.getOffset(0,row);
        for (int col = 0,j=0; col < imi.cols; col++) {
            int sample = db.getElem(elem++);
            line.scanline[j++] =  (sample & 0xFF0000)>>16; // R
            line.scanline[j++] =  (sample & 0xFF00)>>8; // G
            line.scanline[j++] =  (sample & 0xFF); // B
            line.scanline[j++] =  (((sample & 0xFF000000)>>24)&0xFF); // A
        }
        pngw.writeRow(line, row);
    }
    pngw.end();
}

关于java - 在 Java 中以较小的磁盘大小写入 PNG 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15598872/

相关文章:

python - 生成小于 10KB 且不丢失比例的图像缩略图

C# aforge 颜色检测

java - Android 闹钟管理器设置在特定时间重复

java - 选择选项后页面自动刷新

c# - 为什么 Bitmap.Save 会改变图像的大小?

python - Django 图片上传总是失败,表单永远无效

Python OpenCV 如何在矩形内绘制图像的矩形中心和裁剪图像?

java - 在 java 应用程序和 android 应用程序中使用相同的 java 类时的两种不同格式

java - 在 JUnit 测试类中使用随机值

python - 使用 90° 滚动变换和重新映射等距柱状图像