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

时间:2015-02-06 14:15:33

标签: java pdf itext

我正在使用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吗?

1 个答案:

答案 0 :(得分:3)

你已经从我的书中复制/粘贴了一个例子,但似乎你没有读过这本书,也没有真正尝试过这个例子。你说“我不想减少尺寸,但要改变DPI。”

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

img.scaleAbsolute(400, 300);

此图像需要55332个字节(绿点)。

enter image description here

注意我们可以很容易地计算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乘300点。

这是完全您在问题中复制/粘贴的ResizeImage示例中所执行的操作。我们不要看看里面:

enter image description here

图像仍然可以测量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)。您已成功压缩图像。

简而言之:你误解了你正在使用的例子。你说它没有达到你想要的效果。我可以证明它确实存在; - )

根据您在此答案中发布的评论,您似乎会混淆图像分辨率,图像压缩等概念。为了消除这种混淆,您应该阅读以下问题和答案: