如何在不调整图像大小的情况下压缩PDF?

How to compress PDF without resizing image?

我正在使用 iText 压缩现有的 pdf。 我正在使用 iText in Action book(第二版)中的示例:

package part4.chapter16;

import java.awt.Graphics2D;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;

import javax.imageio.ImageIO;

import com.itextpdf.text.DocumentException;
import com.itextpdf.text.pdf.PRStream;
import com.itextpdf.text.pdf.PdfName;
import com.itextpdf.text.pdf.PdfNumber;
import com.itextpdf.text.pdf.PdfObject;
import com.itextpdf.text.pdf.PdfReader;
import com.itextpdf.text.pdf.PdfStamper;
import com.itextpdf.text.pdf.parser.PdfImageObject;

public class ResizeImage {

    /** The resulting PDF file. */
    public static String RESULT = "results/part4/chapter16/resized_image.pdf";
    /** The multiplication factor for the image. */
    public static float FACTOR = 0.5f;

    /**
     * Manipulates a PDF file src with the file dest as result
     * @param src the original PDF
     * @param dest the resulting PDF
     * @throws IOException
     * @throws DocumentException 
     */
    public void manipulatePdf(String src, String dest) throws IOException, DocumentException {
        PdfName key = new PdfName("ITXT_SpecialId");
        PdfName value = new PdfName("123456789");
        // Read the file
        PdfReader reader = new PdfReader(SpecialId.RESULT);
        int n = reader.getXrefSize();
        PdfObject object;
        PRStream stream;
        // Look for image and manipulate image stream
        for (int i = 0; i < n; i++) {
            object = reader.getPdfObject(i);
            if (object == null || !object.isStream())
                continue;
            stream = (PRStream)object;
            if (value.equals(stream.get(key))) {
                PdfImageObject image = new PdfImageObject(stream);
                BufferedImage bi = image.getBufferedImage();
                if (bi == null) continue;
                int width = (int)(bi.getWidth() * FACTOR);
                int height = (int)(bi.getHeight() * FACTOR);
                BufferedImage img = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
                AffineTransform at = AffineTransform.getScaleInstance(FACTOR, FACTOR);
                Graphics2D g = img.createGraphics();
                g.drawRenderedImage(bi, at);
                ByteArrayOutputStream imgBytes = new ByteArrayOutputStream();
                ImageIO.write(img, "JPG", imgBytes);
                stream.clear();
                stream.setData(imgBytes.toByteArray(), false, PRStream.NO_COMPRESSION);
                stream.put(PdfName.TYPE, PdfName.XOBJECT);
                stream.put(PdfName.SUBTYPE, PdfName.IMAGE);
                stream.put(key, value);
                stream.put(PdfName.FILTER, PdfName.DCTDECODE);
                stream.put(PdfName.WIDTH, new PdfNumber(width));
                stream.put(PdfName.HEIGHT, new PdfNumber(height));
                stream.put(PdfName.BITSPERCOMPONENT, new PdfNumber(8));
                stream.put(PdfName.COLORSPACE, PdfName.DEVICERGB);
            }
        }
        // Save altered PDF
        PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(RESULT));
        stamper.close();
        reader.close();
    }

    /**
     * Main method.
     *
     * @param    args    no arguments needed
     * @throws DocumentException 
     * @throws IOException
     */
    public static void main(String[] args) throws IOException, DocumentException {

        new ResizeImage().manipulatePdf(src, dest);
    }
}

以上代码在FACTOR的基础上缩小了图片的大小。我不想缩小尺寸,而是更改 DPI 以减小图像尺寸。 任何帮助将不胜感激。我是 java 的新手。 还有其他一些可用于压缩 PDF 的开源工具吗?

你有copy/pasted我书中的一个例子,但你似乎没有读过这本书,也没有真正尝试过这个例子。你说"I don't want to reduce the dimension, but change DPI."

嗯...这正是我书中的示例所做的!在 SpecialID 示例中,我创建了一个 PDF,其页面大小使用此矩形定义:new Rectangle(400, 300)。这意味着我们有一个 400 x 300 点的页面(见下面屏幕截图中的蓝点)。在此页面上,我添加了一个 500 x 332 像素的 JPG(见红点)。使用以下方法将此图像缩放为 400 x 300 点:

img.scaleAbsolute(400, 300);

此图像占用 55332 字节(绿点)。

注意我们很容易计算出DPI:图片的宽度是400点;那是 5.555 英寸。图片高度为300点;那是 4.166 英寸。宽度的像素数为 500,因此 X 方向的 DPI 为 500 x 72 / 400 或 90 DPI。高度的像素数为 332,因此 DPI 为 332 x 72 / 300 或 79.68 DPI。

您想通过降低分辨率来减少 JPG 的字节数。但是:您希望图像的大小保持不变:它仍然必须覆盖 400 x 300 点。

正是您在问题中copy/pasted的ResizeImage示例中所做的。一起来看看吧:

图像的尺寸仍为 400 x 300 点(这就是您想要的,不是吗?)但分辨率已大幅下降:图像现在为 250 x 166 像素而不是 500 x 332 像素而不改变其尺寸页面大小!

现在让我们计算新的 DPI:在 X 方向上,我们有 250 x 72 / 400。即 45 DPI。在 Y 方向,我们有 166 x 72 / 300。那是 39.84 DPI。 这正好是我们之前 DPI 的一半!这是巧合吗?当然不是!这就是我们使用的 FACTOR。 (这都是简单的数学问题。)

由于降低了分辨率,图像现在只占用 5343 字节(而不是原来的 55332)。您已成功压缩图像。

简而言之:您误解了您使用的示例。你说它没有达到你想要的。我可以证明它确实如此 ;-)

从您对此答案发表的评论来看,您似乎混淆了图像分辨率、图像压缩等概念。要消除这种混淆,您应该阅读以下问题和答案:

  • Re-sizing an image without losing quality
  • Java - resize image without losing quality
  • Increasing Resolution and Reducing Size of an Image in Java
  • Programatically Reducing JPEG file size
  • decrease image resolution in java
  • ...