赞
踩
这篇文章,主要介绍使用itext7将多个PDF文件、图片合并成一个PDF文件,图片旋转、图片缩放。
目录
我这里使用的是itext-core7.1.16版本,只需要引入一个itext-core依赖即可,因为这个依赖里面已经给我们引入了itext所需要的依赖。
- <!-- 引入 itext7-core 依赖 -->
- <dependency>
- <groupId>com.itextpdf</groupId>
- <artifactId>itext7-core</artifactId>
- <version>7.1.16</version>
- <type>pom</type>
- </dependency>
最简单的合并方式,那就是读取两个PDF文件,然后将其合并成一个新的PDF文件,保存到服务器上面之后,在将这个新的PDF文件和下一个待合并的PDF文件进行合并,以此类推,最终可以得到一个完整的PDF文件,但是这种方式缺点在于,每一次合并之后,都需要新生成一个PDF文件,并且下一次合并之后,还要再读取这个PDF文件,这就会导致多次读取文件的过程,效率不是很理想。
这篇文章,我主要是将PDF作为字节数组读取到内存里面,然后在内存中合并两个PDF的字节数据,这样可以减少读取和生成PDF文件的次数,执行效率方面也就会更加好一些了,合并两个PDF字节数组的方法如下所示:
- /**
- * 基于内存中的字节数组进行PDF文档的合并
- * @param firstPdf 第一个PDF文档
- * @param secondPdf 第二个PDF文档
- */
- private static byte[] mergePdfBytes(byte[] firstPdf, byte[] secondPdf) throws IOException {
- if (firstPdf != null && secondPdf != null) {
- // 创建字节数组,基于内存进行合并
- ByteArrayOutputStream baos = new ByteArrayOutputStream();
- PdfDocument destDoc = new PdfDocument(new PdfWriter(baos));
- // 合并的pdf文件对象
- PdfDocument firstDoc = new PdfDocument(new PdfReader(new ByteArrayInputStream(firstPdf)));
- PdfDocument secondDoc = new PdfDocument(new PdfReader(new ByteArrayInputStream(secondPdf)));
- // 合并对象
- PdfMerger merger = new PdfMerger(destDoc);
- merger.merge(firstDoc, 1, firstDoc.getNumberOfPages());
- merger.merge(secondDoc, 1, secondDoc.getNumberOfPages());
- // 关闭文档流
- merger.close();
- firstDoc.close();
- secondDoc.close();
- destDoc.close();
- return baos.toByteArray();
- }
- return null;
- }
合并PDF文件的时候,有些PDF文件可能是网络上的,也有些是本地磁盘上的,所以这里需要做下判断,如果是网络上的PDF文件,则需要首先访问网络,再将其保存到字节数组里面,如果是本地磁盘文件,则需要读取本地文件。
- /**
- * 将pdf文档转换成字节数组
- * @param pdf PDF文档路径
- * @return 返回对应PDF文档的字节数组
- */
- private static byte[] getPdfBytes(String pdf) throws Exception {
- ByteArrayOutputStream out = new ByteArrayOutputStream();
- InputStream is;
- if (pdf.startsWith("http://") || pdf.startsWith("https://")) {
- is = new URL(pdf).openStream();
- } else {
- is = new FileInputStream(pdf);
- }
- byte[] data = new byte[2048];
- int len;
- while ((len = is.read(data)) != -1) {
- out.write(data, 0, len);
- }
- return out.toByteArray();
- }
合并PDF时候,直接传递需要合并的PDF文件路径就可以啦,调用下面方法,就可以完成合并。
- /**
- * 将给定List集合中的pdf文档,按照顺序依次合并,生成最终的目标PDF文档
- * @param pdfPathLists 待合并的PDF文档路径集合,可以是本地PDF文档,也可以是网络上的PDF文档
- * @param destPath 目标合并生成的PDF文档路径
- */
- public static boolean mergeMultiplePdfs(List<String> pdfPathLists, String destPath) {
- try {
- int size = pdfPathLists.size();
- byte[] pdfData = getPdfBytes(pdfPathLists.get(0));
- for (int i = 1; i < size; i++) {
- pdfData = mergePdfBytes(pdfData, getPdfBytes(pdfPathLists.get(i)));
- }
- if (pdfData != null) {
- FileOutputStream fis = new FileOutputStream(destPath);
- fis.write(pdfData);
- fis.close();
- }
- return true;
- } catch (Exception e) {
- logger.error("合并PDF异常:", e);
- }
- return false;
- }
如何将图片也一起合并到PDF文件里面呢???这里我是将图片直接添加到PDF文件的空白页面中实现的,一张图片占据一个页面,当然,你也可以设置显示在相同页面,超过之后页面高度之后,图片会自动显示到下一个页面。
- /**
- * 将给定集合中的图片合并到一个pdf文档里面
- * @param imagePathList 图片路径集合
- * @param destPath 合并之后的PDF文档
- */
- public static boolean mergeImagesToPdf(List<String> imagePathList, String destPath) {
- try {
- PdfDocument pdfDocument = new PdfDocument(new PdfWriter(destPath));
- Document document = new Document(pdfDocument);
- if (imagePathList != null && imagePathList.size() > 0) {
- int size = imagePathList.size();
- for (int i = 0; i < size; i++) {
- String imgPath = imagePathList.get(i);
- ImageData imageData;
- if (imgPath.startsWith("http://") || imgPath.startsWith("https://")) {
- imageData = ImageDataFactory.create(new URL(imgPath));
- } else {
- imageData = ImageDataFactory.create(imgPath);
- }
- Image image = new Image(imageData);
- /*
- 设置旋转的弧度值,默认是逆时针旋转的。
- 弧度、角度换算公式:
- 1° = PI / 180°
- 1 rad = 180° / PI
- */
- image.setRotationAngle(- Math.PI / 2); // 顺时针旋转90°
- // 设置图片自动缩放,即:图片宽高自适应
- image.setAutoScale(true);
- document.add(image);
- if (i != size - 1) {
- // 最后一页不需要新增空白页
- document.add(new AreaBreak(AreaBreakType.NEXT_PAGE));
- }
- }
- }
- pdfDocument.close();
- return true;
- } catch (Exception e) {
- logger.error("合并图片到PDF异常:", e);
- }
- return false;
- }
在某些需求下,你可以想某个图片竖向摆放、某些图片横向摆放,那么这个时候,就可以调用itext7中【Image】图片对象的【setRotationAngle()】方法,对其进行旋转,需要注意的是:setRotationAngle方法设置的旋转弧度,而不是旋转角度,并且它是逆时针旋转的。弧度和角度之间有一个转换公式,如下所示:
- package com.gitcode.itext.util;
-
- import com.itextpdf.io.image.ImageData;
- import com.itextpdf.io.image.ImageDataFactory;
- import com.itextpdf.kernel.pdf.PdfDocument;
- import com.itextpdf.kernel.pdf.PdfReader;
- import com.itextpdf.kernel.pdf.PdfWriter;
- import com.itextpdf.kernel.utils.PdfMerger;
- import com.itextpdf.layout.Document;
- import com.itextpdf.layout.element.AreaBreak;
- import com.itextpdf.layout.element.Image;
- import com.itextpdf.layout.property.AreaBreakType;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
-
- import java.io.*;
- import java.net.URL;
- import java.util.List;
-
- /**
- * @version 1.0.0
- * @Date: 2023/10/04 10:07
- * @Author ZhuYouBin
- * @Description: PDF工具类【基于 itext7 组件实现】
- */
- public class PDFUtil {
- private static final Logger logger = LoggerFactory.getLogger(PDFUtil.class);
-
- /**
- * 将给定List集合中的pdf文档,按照顺序依次合并,生成最终的目标PDF文档
- * @param pdfPathLists 待合并的PDF文档路径集合,可以是本地PDF文档,也可以是网络上的PDF文档
- * @param destPath 目标合并生成的PDF文档路径
- */
- public static boolean mergeMultiplePdfs(List<String> pdfPathLists, String destPath) {
- try {
- int size = pdfPathLists.size();
- byte[] pdfData = getPdfBytes(pdfPathLists.get(0));
- for (int i = 1; i < size; i++) {
- pdfData = mergePdfBytes(pdfData, getPdfBytes(pdfPathLists.get(i)));
- }
- if (pdfData != null) {
- FileOutputStream fis = new FileOutputStream(destPath);
- fis.write(pdfData);
- fis.close();
- }
- return true;
- } catch (Exception e) {
- logger.error("合并PDF异常:", e);
- }
- return false;
- }
-
- /**
- * 将给定集合中的图片合并到一个pdf文档里面
- * @param imagePathList 图片路径集合
- * @param destPath 合并之后的PDF文档
- */
- public static boolean mergeImagesToPdf(List<String> imagePathList, String destPath) {
- try {
- PdfDocument pdfDocument = new PdfDocument(new PdfWriter(destPath));
- Document document = new Document(pdfDocument);
- if (imagePathList != null && imagePathList.size() > 0) {
- int size = imagePathList.size();
- for (int i = 0; i < size; i++) {
- String imgPath = imagePathList.get(i);
- ImageData imageData;
- if (imgPath.startsWith("http://") || imgPath.startsWith("https://")) {
- imageData = ImageDataFactory.create(new URL(imgPath));
- } else {
- imageData = ImageDataFactory.create(imgPath);
- }
- Image image = new Image(imageData);
- /*
- 设置旋转的弧度值,默认是逆时针旋转的。
- 弧度、角度换算公式:
- 1° = PI / 180°
- 1 rad = 180° / PI
- */
- image.setRotationAngle(- Math.PI / 2); // 顺时针旋转90°
- // 设置图片自动缩放,即:图片宽高自适应
- image.setAutoScale(true);
- document.add(image);
- if (i != size - 1) {
- // 最后一页不需要新增空白页
- document.add(new AreaBreak(AreaBreakType.NEXT_PAGE));
- }
- }
- }
- pdfDocument.close();
- return true;
- } catch (Exception e) {
- logger.error("合并图片到PDF异常:", e);
- }
- return false;
- }
-
- /**
- * 基于内存中的字节数组进行PDF文档的合并
- * @param firstPdf 第一个PDF文档
- * @param secondPdf 第二个PDF文档
- */
- private static byte[] mergePdfBytes(byte[] firstPdf, byte[] secondPdf) throws IOException {
- if (firstPdf != null && secondPdf != null) {
- // 创建字节数组,基于内存进行合并
- ByteArrayOutputStream baos = new ByteArrayOutputStream();
- PdfDocument destDoc = new PdfDocument(new PdfWriter(baos));
- // 合并的pdf文件对象
- PdfDocument firstDoc = new PdfDocument(new PdfReader(new ByteArrayInputStream(firstPdf)));
- PdfDocument secondDoc = new PdfDocument(new PdfReader(new ByteArrayInputStream(secondPdf)));
- // 合并对象
- PdfMerger merger = new PdfMerger(destDoc);
- merger.merge(firstDoc, 1, firstDoc.getNumberOfPages());
- merger.merge(secondDoc, 1, secondDoc.getNumberOfPages());
- // 关闭文档流
- merger.close();
- firstDoc.close();
- secondDoc.close();
- destDoc.close();
- return baos.toByteArray();
- }
- return null;
- }
-
- /**
- * 将pdf文档转换成字节数组
- * @param pdf PDF文档路径
- * @return 返回对应PDF文档的字节数组
- */
- private static byte[] getPdfBytes(String pdf) throws Exception {
- ByteArrayOutputStream out = new ByteArrayOutputStream();
- InputStream is;
- if (pdf.startsWith("http://") || pdf.startsWith("https://")) {
- is = new URL(pdf).openStream();
- } else {
- is = new FileInputStream(pdf);
- }
- byte[] data = new byte[2048];
- int len;
- while ((len = is.read(data)) != -1) {
- out.write(data, 0, len);
- }
- return out.toByteArray();
- }
- }
- package com.gitcode.itext;
-
- import com.gitcode.itext.util.PDFUtil;
-
- import java.io.FileNotFoundException;
- import java.util.ArrayList;
- import java.util.List;
-
- /**
- * @version 1.0.0
- * @Date: 2023/10/4 11:12
- * @Author ZhuYouBin
- * @Description:
- */
- public class ImageDemo {
- public static void main(String[] args) throws FileNotFoundException {
- // 图片合并之后生成的PDF路径
- String imagePath = "F:\\pdf-demo\\imagePath.pdf";
- List<String> imageList = new ArrayList<>();
- imageList.add("F:\\pdf-demo\\01.jpg");
- imageList.add("F:\\pdf-demo\\02.jpg");
- // 先合并图片
- PDFUtil.mergeImagesToPdf(imageList, imagePath);
-
- // 在合并PDF
- String destPath = "F:\\pdf-demo\\merge.pdf";
- List<String> pdfPath = new ArrayList<>();
- pdfPath.add("F:\\pdf-demo\\demo01.pdf");
- pdfPath.add("F:\\pdf-demo\\demo02.pdf");
- pdfPath.add(imagePath);
- PDFUtil.mergeMultiplePdfs(pdfPath, destPath);
- }
- }
到此,itext7合并PDF文件就介绍完啦。
综上,这篇文章结束了,主要介绍使用itext7将多个PDF文件、图片合并成一个PDF文件,图片旋转、图片缩放。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。