目录
前言
本文为我搜集的根据模板导出PDF的方法整理而来,所以贴了好多帖子的链接。有的方法仅适合特殊的业务场景,可以根据业务需求选择合适的方法。写的不好请轻喷。
思路一:直接导出pdf
使用itext模板导出pdf
思路二:先导出word再转成pdf
1)导出word
2)word转pdf
- docx4j 将 .docx 转 pdf
官方下载地址
- Spire.Doc 实现 word (.doc / .docx)转 pdf(尝试了免费版,可行)
有付费版和免费版,免费版仅支持三页内的 word 转 pdf
- aspose.word 将 word 转 pdf (未尝试)
需要license,破解方法1,破解方法2
最终方案
docx4j spire.doc.free + freemarker
<ol>模板准备
将占位变量名写在模板 testTemplate.docx的对应位置上,用 ${}包起来。
把复制一份副本,将副本 .docx 的后缀名重命名为 .zip。解压后找到 /word/document.xml,编译器打开文件,代码格式化后,对比例如 ${repliedUserId} 的占位参数是否被拆开(如果拆开需手动修改),修改名字为 testDocument.xml。
将源模板 testTemplate.docx 和 testDocument.xml 放到相应位置。
maven依赖- <dependencies>
-
- <dependency>
- <groupId>org.freemarker</groupId>
- <artifactId>freemarker</artifactId>
- <version>2.3.22</version>
- </dependency>
-
- <dependency>
- <groupId>org.docx4j</groupId>
- <artifactId>docx4j-JAXB-Internal</artifactId>
- <version>8.2.4</version>
- </dependency>
- <dependency>
- <groupId>org.docx4j</groupId>
- <artifactId>docx4j-export-fo</artifactId>
- <version>8.2.4</version>
- </dependency>
-
-
- <dependency>
- <groupId>e-iceblue</groupId>
- <artifactId>spire.doc.free</artifactId>
- <version>5.2.0</version>
- </dependency>
- </dependencies>
- <repositories>
- <repository>
- <id>com.e-iceblue</id>
- <name>e-iceblue</name>
- <url>https://repo.e-iceblue.cn/repository/maven-public/</url>
- </repository>
- </repositories>
复制代码 Controller- @PostMapping("/pdfExport")
- public ResponseEntity exportPdf(@RequestParam Map<String, Object> params) {
- try {
- // 查找业务数据
- TestEntity testEntity = testService.querySheet(params);
- // 格式转换时的暂存文件名
- String fileUuid = UUID.randomUUID().toString().replaceAll("-", "");
- String toDocxPath = "E://project//test//ToPDF//word//" + fileUuid + ".docx";
- String toPdfPath = "E://project//test//ToPDF//pdf//" + fileUuid + ".pdf";
- String toXmlPath = "E://project//test//ToPDF//xml//" + fileUuid + ".xml";
- String docxTemplate = "E://project//test//ToPDF//template//testTemplate.docx";
- // .xml转.docx(testDocument.xml表示在项目的相对路径下)
- XmlToDocx.toDocx("testDocument.xml",docxTemplate, toXmlPath, toDocxPath, testEntity);
- // .docx转.pdf
- WordToPdf.docxToPdf(toDocxPath, toPdfPath);
- // 下载pdf并删除本地pdf
- ResponseEntity response = WordToPdf.downloadPdf("这是PDF的名字啊", toPdfPath);
- return response;
- } catch (Exception e) {
- throw new BusinessException("下载PDF失败!" + e.getMessage());
- }
- }
复制代码 XmlToDocx类
[code]import java.io.*;import java.util.Enumeration;import java.util.Map;import java.util.zip.ZipEntry;import java.util.zip.ZipException;import java.util.zip.ZipFile;import java.util.zip.ZipOutputStream;/** * 其实docx属于zip的一种,这里只需要操作word/document.xml中的数据,其他的数据不用动 * * @author * */public class XmlToDocx { /** * * @param xmlTemplate xml的文件名 * @param docxTemplate docx的路径和文件名(.docx模板) * @param xmlTemp 填充完数据的临时xml * @param toFilePath 目标文件名 * @param object 需要动态传入的数据 */ public static void toDocx(String xmlTemplate, String docxTemplate, String xmlTemp, String toFilePath, Object object) { try { // 1.object是动态传入的数据 // 这个地方不能使用FileWriter因为需要指定编码类型否则生成的Word文档会因为有无法识别的编码而无法打开// Writer w1 = new OutputStreamWriter(new FileOutputStream(xmlTemp), "gb2312"); Writer w1 = new OutputStreamWriter(new FileOutputStream(xmlTemp), "utf-8"); // 2.把object中的数据动态由freemarker传给xml XmlTplUtil.process(xmlTemplate, object, w1); // 3.把填充完成的xml写入到docx中 XmlToDocx xtd = new XmlToDocx(); File xmlTempFile = new File(xmlTemp); xtd.outDocx(xmlTempFile, docxTemplate, toFilePath); // 删除临时xml文件 xmlTempFile.delete(); }catch (Exception e) { e.printStackTrace(); } } /** * * @param documentFile 动态生成数据的docunment.xml文件 * @param docxTemplate docx的模板 * @param toFilePath 需要导出的文件路径 * @throws ZipException * @throws IOException */ public void outDocx(File documentFile, String docxTemplate, String toFilePath) throws ZipException, IOException { try { File docxFile = new File(docxTemplate); ZipFile zipFile = new ZipFile(docxFile); Enumeration |