Springboot

您所在的位置:网站首页 pdf合并成一个pdf要钱吗 Springboot

Springboot

2024-07-14 06:53:21| 来源: 网络整理| 查看: 265

文章目录 前言实现依赖引入工具类编写

前言

最近接了一个需求,客户觉得一个合同导出多项类型的pdf数据,不够直接明了,需要将多个pdf文件 合并 成一个pdf。

实现 依赖引入

编写工具类之前,需要先引入对应的依赖库信息。 本次使用的是com.lowagie.itext这个工具pom,完整依赖如下所示:

com.lowagie itext 2.1.7 工具类编写

实现上述的目的,有以下两种方式:

先生成临时pdf文件,再进行文件合并,并生成新的pdf base64编码返回请求端,最后将临时文件删除。直接将base64编码的值,转换为byte字节码文件,直接进行字节码信息的合并。

先来看方式一:

import java.io.*; import com.lowagie.text.Document; import com.lowagie.text.DocumentException; import com.lowagie.text.Rectangle; import com.lowagie.text.pdf.PdfCopy; import com.lowagie.text.pdf.PdfImportedPage; import com.lowagie.text.pdf.PdfReader; import lombok.Cleanup; import org.springframework.util.CollectionUtils; import sun.misc.BASE64Decoder; import sun.misc.BASE64Encoder; import java.io.FileOutputStream; import java.util.ArrayList; import java.util.List; import java.util.UUID; /** * @Title: 需要引入com.lowagie.itext 2.1.7 依赖 * @description: 多个pdf 文件合并成一个pdf文件 **/ public class PdfMergeUtils { /** * 多个pdf文件合并 * @param files 待合并的文件数组 * @param newfile 新的文件全路径信息(带文件名) * @return */ public static boolean mergePdfFiles(String[] files, String newfile) { boolean retValue = false; Document document = null; PdfCopy copy = null; PdfReader reader = null; try { document = new Document(new PdfReader(files[0]).getPageSize(1)); copy = new PdfCopy(document, new FileOutputStream(newfile)); document.open(); for (int i = 0; i document.newPage(); PdfImportedPage page = copy.getImportedPage(reader, j); copy.addPage(page); } reader.close(); } retValue = true; } catch (Exception e) { e.printStackTrace(); } finally { if (reader != null) { reader.close(); } if (copy != null) { copy.close(); } if (document != null) { document.close(); } } return retValue; } /** * 将文件集合信息 ,合并到 newfile文件名的新pdf文件中 * @param files 待合并的pdf文件 * @param newfile 合并后新生成的pdf文件(如果不是全路径,则会生成在项目同级目录下) * @return */ public static boolean mergePdfFiles(List files, String newfile) { if(CollectionUtils.isEmpty(files)){ return Boolean.FALSE; } boolean retValue = false; Document document = null; PdfCopy copy = null; PdfReader reader = null; try { // 以第一个pdf 作为基准 PdfReader pdfReader = new PdfReader(files.get(0)); Rectangle rectangle = pdfReader.getPageSize(1); document = new Document(rectangle); // 以第一个pdf 构建新的pdf文件输出流 copy = new PdfCopy(document, new FileOutputStream(newfile)); document.open(); for (int i = 0; i document.newPage(); // 创建新的pdf页数 PdfImportedPage page = copy.getImportedPage(reader, j); copy.addPage(page); } reader.close(); } retValue = true; } catch (Exception e) { e.printStackTrace(); } finally { if (reader != null) { reader.close(); } if (copy != null) { copy.close(); } if (document != null) { document.close(); } } return retValue; } /** * 将pdf的base64 编码,转换成对应的pdf文件暂存,并返回文件名称集合 * @param base64Lists * @return * @throws Exception */ public static List base64ToFile(List base64Lists) throws Exception { if(CollectionUtils.isEmpty(base64Lists)){ return null; } List returnStrLists = new ArrayList(); BASE64Decoder decoder = new BASE64Decoder(); for (String base64Str : base64Lists) { byte[] fileBytes = decoder.decodeBuffer(base64Str); @Cleanup ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(fileBytes); // 此处代码会导致生成的文件很大 // @Cleanup ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(fileBytes.length); // 生成临时文件 为了保证文件的唯一性,必须名称不同 String fileName = "report_"+UUID.randomUUID().toString()+".pdf"; returnStrLists.add(fileName); // 返回调用方文件名集合 // 暂存文件 @Cleanup OutputStream outputStream = new FileOutputStream(fileName); // int c = 0; // while (c < byteArrayInputStream.read()){ // byteArrayOutputStream.write(fileBytes); // byteArrayOutputStream.writeTo(outputStream); // byteArrayOutputStream.flush(); // } byte[] bytes = new byte[1024]; int len = 0; while ((len = byteArrayInputStream.read(bytes)) > 0){ outputStream.write(bytes,0,len); } } return returnStrLists; } /** * 根据获取到的pdf base64 文件流集合,生成临时文件,并重新读取临时文件,进行pdf的合并。 * @param base64Pdfs 各个合同类型数据pdf文件base64数据流 * @return 新的已合并的pdf文件流的base64 数据值 * @throws Exception */ public static String getNewMergePdfBase64(List base64Pdfs) throws Exception { String newPdfName = null; List fileNameLists = null; try { fileNameLists = base64ToFile(base64Pdfs); newPdfName = "report_"+UUID.randomUUID().toString()+".pdf"; mergePdfFiles(fileNameLists, newPdfName); // 以新的文件名 读取成文件流 并转换base64 返回 @Cleanup FileInputStream fileInputStream = new FileInputStream(newPdfName); return base64ForPdf(fileInputStream); }finally { // 如果临时文件存在,则删除临时文件 if(StringUtils.isNotBlank(newPdfName)){ File file = new File(newPdfName); if (file != null && file.isFile() && file.exists()) { file.delete(); } } if(!CollectionUtils.isEmpty(fileNameLists)){ for (String fileName : fileNameLists) { File file = new File(fileName); if (file != null && file.isFile() && file.exists()) { file.delete(); } } } } } /** *将文件输入流,转换为 base64 返回给请求端 **/ public static String base64ForPdf(InputStream fin) throws Exception { BASE64Encoder encoder = new BASE64Encoder(); BufferedInputStream bin = new BufferedInputStream(fin); ByteArrayOutputStream baos = new ByteArrayOutputStream(); BufferedOutputStream bout = new BufferedOutputStream(baos); byte[] buffer = new byte[1024]; for (int len = bin.read(buffer); len != -1; len = bin.read(buffer)) { bout.write(buffer, 0, len); } bout.flush(); byte[] bytes = baos.toByteArray(); String var11 = encoder.encodeBuffer(bytes).trim(); return var11; } }

方式二就相对简单点,舍去了中间生成各个临时文件的逻辑,直接进行pdf数据流的合并。逻辑如下所示:

/** * pdf 合并操作2 * 直接将各个pdf对应的字节码,进行合并成新的pdf文件数据 * @param base64Pdfs 多个pdf的base64编码文件 * @return */ public static String getNewMergePdfBase64_2(List base64Pdfs) throws Exception { String newPdfName = null; try{ List byteLists = base64ToByte(base64Pdfs); // 将pdf的byte[] 数据生成新的pdf文件 if(CollectionUtils.isEmpty(byteLists)){ return null; } // 生成新的pdf文件 newPdfName = "report_"+UUID.randomUUID().toString()+".pdf"; mergePdfFiles2(byteLists,newPdfName); @Cleanup FileInputStream fileInputStream = new FileInputStream(newPdfName); return base64ForPdf(fileInputStream); }finally { // 如果临时文件存在,则删除临时文件 if(StringUtils.isNotBlank(newPdfName)){ File file = new File(newPdfName); if (file != null && file.isFile() && file.exists()) { file.delete(); } } } } public static void mergePdfFiles2(List bytes, String newFile) { try { // 以第一个pdf作为基础,后面的每页信息逐渐累加 Document document = new Document(new PdfReader(bytes.get(0)).getPageSize(1)); PdfCopy copy = new PdfCopy(document, new FileOutputStream(newFile)); document.open(); for (byte[] aByte : bytes) { PdfReader reader = new PdfReader(aByte); int n = reader.getNumberOfPages(); for (int j = 1; j e.printStackTrace(); } } /** * base62 文件,转byte[] * @param base64Lists 多个pdf的base64 编码 * @return 多个pdf文件的字节码数组集合 */ private static List base64ToByte(List base64Lists) throws Exception { if(CollectionUtils.isEmpty(base64Lists)){ return null; } List returnStrLists = new ArrayList(); BASE64Decoder decoder = new BASE64Decoder(); for (String base64Str : base64Lists) { byte[] fileBytes = decoder.decodeBuffer(base64Str); returnStrLists.add(fileBytes); } return returnStrLists; }

@Cleanup 注解的作用就是代码执行完成后,默认调用对应对象的close(),就好比第一个方法中的finally。



【本文地址】

公司简介

联系我们

今日新闻


点击排行

实验室常用的仪器、试剂和
说到实验室常用到的东西,主要就分为仪器、试剂和耗
不用再找了,全球10大实验
01、赛默飞世尔科技(热电)Thermo Fisher Scientif
三代水柜的量产巅峰T-72坦
作者:寞寒最近,西边闹腾挺大,本来小寞以为忙完这
通风柜跟实验室通风系统有
说到通风柜跟实验室通风,不少人都纠结二者到底是不
集消毒杀菌、烘干收纳为一
厨房是家里细菌较多的地方,潮湿的环境、没有完全密
实验室设备之全钢实验台如
全钢实验台是实验室家具中较为重要的家具之一,很多

推荐新闻


图片新闻

实验室药品柜的特性有哪些
实验室药品柜是实验室家具的重要组成部分之一,主要
小学科学实验中有哪些教学
计算机 计算器 一般 打孔器 打气筒 仪器车 显微镜
实验室各种仪器原理动图讲
1.紫外分光光谱UV分析原理:吸收紫外光能量,引起分
高中化学常见仪器及实验装
1、可加热仪器:2、计量仪器:(1)仪器A的名称:量
微生物操作主要设备和器具
今天盘点一下微生物操作主要设备和器具,别嫌我啰嗦
浅谈通风柜使用基本常识
 众所周知,通风柜功能中最主要的就是排气功能。在

专题文章

    CopyRight 2018-2019 实验室设备网 版权所有 win10的实时保护怎么永久关闭