资讯专栏INFORMATION COLUMN

java根据模板动态生成PDF

liukai90 / 2382人阅读

摘要:一需求说明根据业务需要,需要在服务器端生成可动态配置的文档,方便数据可视化查看。能配置动态的模板,正好解决了样式动态渲染和排版问题。包负责模板之外的额外信息填写,这里主要是页眉页脚的定制。包的画图工具包,目前只有一个线形图。

一、需求说明:根据业务需要,需要在服务器端生成可动态配置的PDF文档,方便数据可视化查看。

二、解决方案:
iText+FreeMarker+JFreeChart生成可动态配置的PDF文档
iText有很强大的PDF处理能力,但是样式和排版不好控制,直接写PDF文档,数据的动态渲染很麻烦。
FreeMarker能配置动态的html模板,正好解决了样式、动态渲染和排版问题。
JFreeChart有这方便的画图API,能画出简单的折线、柱状和饼图,基本能满足需要。

三、实现功能:

</>复制代码

  1. 1、能动态配置PDF文档内容
  2. 2、支持中文字体显示的动态配置
  3. 3、设置自定义的页眉页脚信息
  4. 4、能动态生成业务图片
  5. 5、完成PDF的分页和图片的嵌入

四、主要代码结构说明:

</>复制代码

  1. 1、component包:PDF生成的组件 对外提供的是PDFKit工具类和HeaderFooterBuilder接口,其中PDFKit负责PDF的生成,HeaderFooterBuilder负责自定义页眉页脚信息。
  2. 2、builder包:负责PDF模板之外的额外信息填写,这里主要是页眉页脚的定制。
  3. 3、chart包:JFreeChart的画图工具包,目前只有一个线形图。
  4. 4、test包:测试工具类
  5. 5、util包:FreeMarker等工具类。

五、关键代码说明:

1、模板配置

</>复制代码

  1. ${templateName}

  2. iText官网:${ITEXTUrl}

  3. FreeMarker官网:${freeMarkerUrl}

  4. JFreeChart教程:${JFreeChartUrl}

  5. 列表值:
  6. <#list scores as item>
  7. ${item}

  8. 第二页开始了
  9. 百度图标

  10. 气温变化对比图

2、获取模板内容并填充数据

</>复制代码

  1. /**
  2. * @description 获取模板
  3. */
  4. public static String getContent(String fileName,Object data){
  5. String templatePath=getPDFTemplatePath(fileName);//根据PDF名称查找对应的模板名称
  6. String templateFileName=getTemplateName(templatePath);
  7. String templateFilePath=getTemplatePath(templatePath);
  8. if(StringUtils.isEmpty(templatePath)){
  9. throw new FreeMarkerException("templatePath can not be empty!");
  10. }
  11. try{
  12. Configuration config = new Configuration(Configuration.VERSION_2_3_25);//FreeMarker配置
  13. config.setDefaultEncoding("UTF-8");
  14. config.setDirectoryForTemplateLoading(new File(templateFilePath));//注意这里是模板所在文件夹,不是文件
  15. config.setTemplateExceptionHandler(TemplateExceptionHandler.RETHROW_HANDLER);
  16. config.setLogTemplateExceptions(false);
  17. Template template = config.getTemplate(templateFileName);//根据模板名称 获取对应模板
  18. StringWriter writer = new StringWriter();
  19. template.process(data, writer);//模板和数据的匹配
  20. writer.flush();
  21. String html = writer.toString();
  22. return html;
  23. }catch (Exception ex){
  24. throw new FreeMarkerException("FreeMarkerUtil process fail",ex);
  25. }
  26. }

3、导出模板到PDF文件

</>复制代码

  1. /**
  2. * @description 导出pdf到文件
  3. * @param fileName 输出PDF文件名
  4. * @param data 模板所需要的数据
  5. *
  6. */
  7. public String exportToFile(String fileName,Object data){
  8. String htmlData= FreeMarkerUtil.getContent(fileName, data);//获取FreeMarker的模板数据
  9. if(StringUtils.isEmpty(saveFilePath)){
  10. saveFilePath=getDefaultSavePath(fileName);//设置PDF文件输出路径
  11. }
  12. File file=new File(saveFilePath);
  13. if(!file.getParentFile().exists()){
  14. file.getParentFile().mkdirs();
  15. }
  16. FileOutputStream outputStream=null;
  17. try{
  18. //设置输出路径
  19. outputStream=new FileOutputStream(saveFilePath);
  20. //设置文档大小
  21. Document document = new Document(PageSize.A4);//IText新建PDF文档
  22. PdfWriter writer = PdfWriter.getInstance(document, outputStream);//设置文档和输出流的关系
  23. //设置页眉页脚
  24. PDFBuilder builder = new PDFBuilder(headerFooterBuilder,data);
  25. builder.setPresentFontSize(10);
  26. writer.setPageEvent(builder);
  27. //输出为PDF文件
  28. convertToPDF(writer,document,htmlData);
  29. }catch(Exception ex){
  30. throw new PDFException("PDF export to File fail",ex);
  31. }finally{
  32. IOUtils.closeQuietly(outputStream);
  33. }
  34. return saveFilePath;
  35. }

4、测试工具类

</>复制代码

  1. public String createPDF(Object data, String fileName){
  2. //pdf保存路径
  3. try {
  4. //设置自定义PDF页眉页脚工具类
  5. PDFHeaderFooter headerFooter=new PDFHeaderFooter();
  6. PDFKit kit=new PDFKit();
  7. kit.setHeaderFooterBuilder(headerFooter);
  8. //设置输出路径
  9. kit.setSaveFilePath("/Users/fgm/Desktop/pdf/hello.pdf”);//设置出书路径
  10. String saveFilePath=kit.exportToFile(fileName,data);
  11. return saveFilePath;
  12. } catch (Exception e) {
  13. log.error("PDF生成失败{}", ExceptionUtils.getFullStackTrace(e));
  14. return null;
  15. }
  16. }

</>复制代码

  1. public static void main(String[] args) {
  2. ReportKit360 kit=new ReportKit360();
  3. TemplateBO templateBO=new TemplateBO();//配置模板数据
  4. templateBO.setTemplateName("Hello iText! Hello freemarker! Hello jFreeChart!");
  5. templateBO.setFreeMarkerUrl("http://www.zheng-hang.com/chm/freemarker2_3_24/ref_directive_if.html");
  6. templateBO.setITEXTUrl("http://developers.itextpdf.com/examples-itext5");
  7. templateBO.setJFreeChartUrl("http://www.yiibai.com/jfreechart/jfreechart_referenced_apis.html");
  8. templateBO.setImageUrl("https://ss0.bdstatic.com/5aV1bjqh_Q23odCf/static/superman/img/logo/bd_logo1_31bdc765.png");
  9. List scores=new ArrayList();
  10. scores.add("90");
  11. scores.add("95");
  12. scores.add("98");
  13. templateBO.setScores(scores);
  14. List lineList=getTemperatureLineList();
  15. TemperatureLineChart lineChart=new TemperatureLineChart();
  16. String picUrl=lineChart.draw(lineList,0);//自定义的数据画图
  17. templateBO.setPicUrl(picUrl);
  18. String path= kit.createPDF(templateBO,"hello.pdf");
  19. System.out.println(path);
  20. }

六、生成效果图:

七、项目完整代码

</>复制代码

  1. 1、github地址:https://github.com/superad/pdf-kit
  2. 2、项目git地址:git@github.com:superad/pdf-kit.git

八、遇到的坑:

1、FreeMarker配置模板文件样式,在实际PDF生成过程中,可能会出现一些不一致的情形,目前解决方法,就是换种方式调整样式。

2、字体文件放在resource下,在打包时会报错,运行mvn -X compile 会看到详细错误:
这是字体文件是二进制的,而maven项目中配置了资源文件的过滤,不能识别二进制文件导致的,
plugins中增加下面这个配置就好了:

</>复制代码

  1. src/main/resources
  2. true
  3. org.apache.maven.plugins
  4. maven-resources-plugin
  5. 2.7
  6. UTF-8
  7. ttf

3、PDF分页配置:

</>复制代码

  1. 在ftl文件中,增加分页标签:

九、 完整maven配置:

</>复制代码

  1. com.itextpdf
  2. itextpdf
  3. 5.4.2
  4. com.itextpdf.tool
  5. xmlworker
  6. 5.4.1
  7. com.itextpdf
  8. itext-asian
  9. 5.2.0
  10. org.xhtmlrenderer
  11. flying-saucer-pdf
  12. 9.0.3
  13. org.freemarker
  14. freemarker
  15. 2.3.26-incubating
  16. jfreechart
  17. jfreechart
  18. 1.0.0
  19. ch.qos.logback
  20. logback-core
  21. 1.0.13
  22. ch.qos.logback
  23. logback-classic
  24. 1.0.13
  25. ch.qos.logback
  26. logback-access
  27. 1.0.13
  28. org.slf4j
  29. slf4j-api
  30. 1.7.5
  31. org.slf4j
  32. log4j-over-slf4j
  33. 1.7.21
  34. com.google.guava
  35. guava
  36. 20.0
  37. org.projectlombok
  38. lombok
  39. 1.14.8
  40. org.apache.commons
  41. commons-io
  42. 1.3.2
  43. commons-lang
  44. commons-lang
  45. 2.6
  46. javax.servlet
  47. servlet-api
  48. 2.5

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/69900.html

相关文章

  • java根据模板动态生成PDF

    摘要:一需求说明根据业务需要,需要在服务器端生成可动态配置的文档,方便数据可视化查看。能配置动态的模板,正好解决了样式动态渲染和排版问题。包负责模板之外的额外信息填写,这里主要是页眉页脚的定制。包的画图工具包,目前只有一个线形图。 一、需求说明:根据业务需要,需要在服务器端生成可动态配置的PDF文档,方便数据可视化查看。 二、解决方案:iText+FreeMarker+JFreeChart生...

    layman 评论0 收藏0
  • spring boot itextPdf根据模板生成pdf文件

    摘要:在开发一些平台中会遇到将数据库中的数据渲染到模板文件中的场景,用完全动态生成文件的太过复杂,通过可以比较简单的完成数据渲染工作模板的表单域数据需定义名称获取输出流下载相关申请表下载设置响应设置响应文件名称申请表设置文件名称获取输出流 在开发一些平台中会遇到将数据库中的数据渲染到PDF模板文件中的场景,用itextPdf完全动态生成PDF文件的太过复杂,通过itextPdf/AcroFi...

    fuyi501 评论0 收藏0
  • Spring Boot集成JasperReports生成PDF文档

    摘要:由于工作需要,要实现后端根据模板动态填充数据生成文档,通过技术选型,使用来设计模板,结合工具库来调用渲染生成文档。 由于工作需要,要实现后端根据模板动态填充数据生成PDF文档,通过技术选型,使用Ireport5.6来设计模板,结合JasperReports5.6工具库来调用渲染生成PDF文档。本人文采欠缺,写作能力差,下面粗略的介绍其使用步骤,若有不对的地方,望大家莫喷,谢谢! 一、使...

    Miracle 评论0 收藏0
  • 常用的6款Java开源报表制作工具

    摘要:本文为大家推荐款常用的开源报表制作工具,供开发者学习参考。一个基于的开源报表工具,它可以在环境下像其他报表工具一样来制作报表,支持和文件输出格式,是当前开发者最常用的报表工具。使用开发的,并使用作为报表生成引擎。 本文为大家推荐6款常用的Java开源报表制作工具,供开发者学习、参考。 1.Aspose.Cells for JasperReports一个基于Java的开源报表工具,它可以...

    QLQ 评论0 收藏0

发表评论

0条评论

liukai90

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<