主要观点总结
本文介绍了关于社群交流、开源项目和PDF生成的技术内容,包括不同的PDF生成方案、技术方案的优缺点,以及相关的代码实现。文章提供了多个技术方案的详细解释,包括使用的工具、流程、优缺点,并给出了代码示例。
关键观点总结
关键观点1: 社群交流
文章提到一个社群,提供一对一交流、面试小册、简历优化、求职解惑等服务。
关键观点2: 开源项目介绍
文章介绍了一些开源项目,包括后端支持单体和微服务架构、功能涵盖RBAC权限、SaaS多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号、CRM等等。还提供了项目的地址和教程链接。
关键观点3: PDF生成方案
文章介绍了四种不同的PDF生成方案:1)itext表单填充;2)freemarker + doc4J 基于Word生成PDF;3)freemarker + aspose-words导出PDF;4)html + freemarker + itextpdf(html2pdf)。每种方案都详细解释了使用方式、优缺点,并给出了相关的代码示例。
关键观点4: 核心思路与工具变化
文章强调解决问题的核心思路一直没有变化,只是使用的工具在变化。需要保持思路清晰。
关键观点5: HtmlToPdfUtil工具类
文章提供了一个HtmlToPdfUtil工具类,用于将Html转换为Pdf。这个类包含了多个方法,如htmlToPdf、xmlFormat等,涉及到了模板加载、字体处理、模板渲染等功能。
正文
👉 这是一个或许对你有用的社群
🐱 一对一交流/面试小册/简历优化/求职解惑,欢迎加入「芋道快速开发平台」知识星球。下面是星球提供的部分资料:
👉这是一个或许对你有用的开源项目
国产 Star 破 10w+ 的开源项目,前端包括管理后台 + 微信小程序,后端支持单体和微服务架构。
功能涵盖 RBAC 权限、SaaS 多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号、CRM 等等功能:
- Boot 仓库:https://gitee.com/zhijiantianya/ruoyi-vue-pro
- Cloud 仓库:https://gitee.com/zhijiantianya/yudao-cloud
- 视频教程:https://doc.iocoder.cn
【国内首批】支持 JDK 21 + SpringBoot 3.2.2、JDK 8 + Spring Boot 2.7.18 双版本
来源:Java知音
这个文档太复杂了,还要导出pdf?
废话不多说直接进入正题,首先分析生成pdf场景及生成内容,考虑复用性和维护难度是我们当前开发工作的第一要务!
下面是调研的几个主要方案:
使用方式:
- itext表单填充方案是以pdf作为基础模板,通过在pdf中嵌入表单元素组件的方式(需要使用pdf编辑工具),最后由程序进行数据填充并另存为pdf结果。
方案优缺点:
- 缺点:原始模板变化需要重新生成pdf,重新编辑表单元素;不支持列表填充数据。
基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
- 项目地址:https://github.com/YunaiV/ruoyi-vue-pro
- 视频教程:https://doc.iocoder.cn/video/
使用方式:
- 首先将调整好格式的原始 word 导出为 XML 格式,编辑 XML 模板中需要填充元素的位置,最后由程序处理先由freemarker模板工具替换元素内容,再使用doc4J进行pdf导出。
方案优缺点:
- 缺点:XML 格式的word真的有够复杂,想要在此模板上调整样式真的难上加难;由于系统不支持的原因需要导入中文字体库;doc4J 部分 doc 元素不支持(例如直线),导出格式差异较大。
这可能是由于doc4J迭代问题无法保证新元素的支持,导出结果比较奔放。。。
基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
- 项目地址:https://github.com/YunaiV/yudao-cloud
- 视频教程:https://doc.iocoder.cn/video/
使用方式:
- 类似于 freemarker+doc4J 方式,同样需要编辑XML,导出格式相较doc4J而言有极大提升。
方案优缺点:
- 优点:通用性强,基于模板引擎功能强大,无需手工管理字体(macOS),代码简单,导出格式与模板基本无差异。
- 缺点:需要编辑 XML 模板;该方案不是免费版(当然有大神)。
受限于调试前期需要的修修改改,模板能给人整吐了,所以才有了下一个方案。
使用方式:
- 翻译 word 为 html 页面(当然就是手写啦,还原度很重要!),html中模板元素插入(文字填充、列表循环 freemarker 支持的全都能写),最后由程序处理先由freemarker模板工具替换元素内容,再使用html2pdf进行pdf导出。
方案优缺点:
- 优点:可维护性相较与上面方案都有极大提升(调试可见性,动态替换生效);通用性强,基于模板引擎功能强大;导出格式可控性较强;
这个方案是综合以上多次踩坑的结果,结果是显而易见的。
浅浅来一点代码,省的大家到处找
<dependencies>
<dependency>
<groupId>com.itextpdfgroupId>
<artifactId>itextpdfartifactId>
<version>5.5.13version>
dependency>
<dependency>
<groupId>com.itextpdfgroupId>
<artifactId>html2pdfartifactId>
<version>3.0.3version>
dependency>
<dependency>
<groupId>org.xhtmlrenderergroupId>
<artifactId>flying-saucer-pdf-itext5artifactId>
<version>9.0.3version>
dependency>
<dependency>
<groupId>binarta.ossgroupId>
<artifactId>groovy-template-enginex-freemarkerartifactId>
<version>0.1.3version>
dependency>
<dependencies>
个人以为自己的代码会自解释,就不贴太多注释了
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.map.MapUtil;
import com.google.common.collect.Lists;
import com.itextpdf.text.pdf.BaseFont;
import freemarker.cache.ByteArrayTemplateLoader;
import freemarker.template.Configuration;
import freemarker.template.Template;
import lombok.SneakyThrows;
import org.xhtmlrenderer.pdf.ITextFontResolver;
import org.xhtmlrenderer.pdf.ITextRenderer;
import java.io.BufferedWriter;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.StringWriter;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.function.Supplier;
public class HtmlToPdfUtil {
private static final ByteArrayTemplateLoader TEMPLATE_LOADER = new ByteArrayTemplateLoader();
// 导入需要字体库的位置哦;simsun 为 宋体
public static final String FRONT_PATH = "/usr/share/fonts/simsun.ttc";
/**
* 看明白的话只用这个方法就够
*/
public static ByteArrayOutputStream htmlToPdf(String templateName, Supplier<byte[]> loadTemplateSupplier, Map modeViewMap) {
String html = xmlFormat(templateName, loadTemplateSupplier, modeViewMap);
return htmlToPdf(html);
}
@SneakyThrows
public static ByteArrayOutputStream htmlToPdf(String htmlStr) {
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
ITextRenderer renderer = new ITextRenderer();
renderer.setDocumentFromString(htmlStr);
ITextFontResolver resolver = renderer.getFontResolver();
//添加字体,解决中文不显示的问题
resolver.addFont(FRONT_PATH, BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
renderer.layout();
renderer.createPDF(outputStream);
return outputStream;
}
public static String xmlFormat(String templateName, Supplier<byte[]> loadTemplateSupplier, Map modeViewMap) {
if (Objects.isNull(TEMPLATE_LOADER.findTemplateSource(templateName))) {
synchronized (TEMPLATE_LOADER) {
if (Objects.isNull(TEMPLATE_LOADER.findTemplateSource(templateName))) {
TEMPLATE_LOADER.putTemplate(templateName, loadTemplateSupplier.get());
}
}
}
return xmlFormat(templateName, modeViewMap);
}
@SneakyThrows
public static String xmlFormat(String templateName, Map modeViewMap) {
Configuration cfg = new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS);
// 指定FreeMarker模板文件的位置
cfg.setTemplateLoader(TEMPLATE_LOADER);
// 设置模板的编码格式
cfg.setEncoding(Locale.CHINA, Charset.defaultCharset().name());
// 获取模板文件 template
Template template = cfg.getTemplate(templateName, Charset.defaultCharset().name());
StringWriter stringWriter = new StringWriter();
BufferedWriter writer = new BufferedWriter(stringWriter);
template.process(modeViewMap, writer);
return stringWriter.toString();
}
}
解决这个问题的核心思路方案其实一直没变,变化的只是工具,一定要思路清晰!
欢迎加入我的知识星球,全面提升技术能力。
👉 加入方式,“长按”或“扫描”下方二维码噢:
星球的内容包括:项目实战、面试招聘、源码解析、学习路线。
文章有帮助的话,在看,转发吧。
谢谢支持哟 (*^__^*)