java集成itextpdf实现通过pdf模板填充数据生成pdf

2023-10-08 11:40

本文主要是介绍java集成itextpdf实现通过pdf模板填充数据生成pdf,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、制作pdf模板
    • 1.1、使用excel制作一个表格
    • 1.2、转成pdf
    • 1.3、设置表单域
    • 1.4、最终模版效果
  • 二、引入POM依赖
  • 三、代码实现
    • 3.1、工具类
    • 3.2、实体对象
    • 3.3、Controller


一、制作pdf模板

1.1、使用excel制作一个表格

在这里插入图片描述

1.2、转成pdf

我采用的是pdfelement 官网地址需要付费或者自行破解,也可以使用其他pdf编辑器。

在这里插入图片描述

1.3、设置表单域

在这里插入图片描述

1.4、最终模版效果

在这里插入图片描述

二、引入POM依赖

<!--pdf处理--><dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.13</version></dependency><dependency><groupId>com.itextpdf</groupId><artifactId>itext-asian</artifactId><version>5.2.0</version></dependency>

三、代码实现

将制作好的pdf模板放入项目resources/pdf目录下,如图

在这里插入图片描述

3.1、工具类

import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.pdf.AcroFields;
import com.itextpdf.text.pdf.BaseFont;
import com.itextpdf.text.pdf.PdfReader;
import com.itextpdf.text.pdf.PdfStamper;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.io.IOUtils;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.util.ResourceUtils;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;/*** pdf相关工具类* 注意事项:* 1、时间类型需要转成字符串* 2、只支持对象嵌对象,两层结构;对象嵌对象再嵌对象,第三层对象无法解析,如果遇到这种情况需要写成递归或解析第三次。** @author zero*/
@Slf4j
public class PdfUtil {private static boolean isPrimitiveOrWrapper(Class<?> clazz) {return clazz.isPrimitive() || clazz.getName().startsWith("java.lang");}private static Map<String, String> turnMap(Object object) {Map<String, Object> stringObjectMap = BeanUtil.beanToMap(object);Map<String, String> map = new HashMap<>(stringObjectMap.size() * 2);// 打印输出属性名称和属性值for (Map.Entry<String, Object> entry : stringObjectMap.entrySet()) {String key = entry.getKey();Object value = entry.getValue();if (ObjectUtil.isNotEmpty(value)) {//基本类型和封装类型if (isPrimitiveOrWrapper(value.getClass())) {map.put(key, String.valueOf(value));} else {//其他类型if (value instanceof List) {List<Object> list = (List) value;for (int i = 0; i < list.size(); i++) {Object o = list.get(i);Map<String, Object> stringObjectMap1 = BeanUtil.beanToMap(o);for (Map.Entry<String, Object> entry1 : stringObjectMap1.entrySet()) {String key1 = entry1.getKey();Object value1 = entry1.getValue();map.put(StrUtil.format("{}.{}{}", key, key1, i), String.valueOf(value1));}}} else {Map<String, Object> stringObjectMap1 = BeanUtil.beanToMap(value);for (Map.Entry<String, Object> entry1 : stringObjectMap1.entrySet()) {String key1 = entry1.getKey();Object value1 = entry1.getValue();map.put(StrUtil.format("{}.{}", key, key1), String.valueOf(value1));}}}}}return map;}public static void generatePdf(HttpServletRequest request,HttpServletResponse response,String templateName, Object object,boolean download) {try (OutputStream responseOutputStream = response.getOutputStream();ByteArrayOutputStream fileOut = new ByteArrayOutputStream()) {//模板在项目中的位置Resource resource = new PathMatchingResourcePatternResolver().getResource(ResourceUtils.CLASSPATH_URL_PREFIX + "pdf/" + templateName);PdfReader reader = new PdfReader(resource.getInputStream());PdfStamper ps = new PdfStamper(reader, fileOut);
//            BaseFont bf = BaseFont.createFont("STSong-Light","UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);BaseFont bf = BaseFont.createFont(BaseFont.HELVETICA_BOLD, BaseFont.WINANSI, BaseFont.EMBEDDED);ArrayList<BaseFont> fontList = new ArrayList<>();fontList.add(bf);//取出报表模板中的所有字段AcroFields fields = ps.getAcroFields();fields.setSubstitutionFonts(fontList);PdfUtil.fillData(fields, PdfUtil.turnMap(object));//必须要调用这个,否则文档不会生成的  如果为false那么生成的PDF文件还能编辑,一定要设为trueps.setFormFlattening(true);ps.close();if (download) {writerFile(request, response, templateName, false);}fileOut.writeTo(responseOutputStream);} catch (Exception e) {log.error("pdf生成异常:", e);throw new RuntimeException("操作异常请联系管理员!");}}/*** 填充数据** @param fields* @param data* @throws IOException* @throws DocumentException*/private static void fillData(AcroFields fields, Map<String, String> data) throws IOException, DocumentException {Map<String, AcroFields.Item> formFields = fields.getFields();for (String key : data.keySet()) {if (formFields.containsKey(key)) {String value = data.get(key);// 为字段赋值,注意字段名称是区分大小写的fields.setField(key, value);}}}/*** 写出文件** @param request* @param response* @param fileName* @param deleteOnExit 是否需要删除本地文件*/private static void writerFile(HttpServletRequest request,HttpServletResponse response,String fileName,boolean deleteOnExit) throws IOException {File file = new File("/" + fileName);file.createNewFile();response.setCharacterEncoding(request.getCharacterEncoding());response.setContentType("application/pdf");try (FileInputStream fis = new FileInputStream(file);) {//这里主要防止下载的PDF文件名乱码response.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode(file.getName(), "UTF-8"));IOUtils.copy(fis, response.getOutputStream());response.flushBuffer();if (deleteOnExit) {file.deleteOnExit();}} catch (Exception e) {log.error("pdf生成异常1:", e);throw new RuntimeException("操作异常1请联系管理员!");}}
}

3.2、实体对象


import lombok.Data;import java.util.List;/*** @author zero*/
@Data
public class TestDto {private String name;private String birthday;private Other other;private List<Other> otherList;@Datapublic static class Other {private String career;private String company;}}

3.3、Controller

import io.swagger.annotations.Api;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.ArrayList;
import java.util.List;/*** 测试 控制层** @author zero*/
@RestController
@RequestMapping("test")
@Api(tags = "测试")
public class TestController {@GetMapping("/pdf")public void pdf(HttpServletRequest request, HttpServletResponse response) {TestDto testDto = new TestDto();testDto.setName("张三");testDto.setBirthday("2020-12-12");TestDto.Other other = new TestDto.Other();other.setCareer("码农");other.setCompany("字节不跳动");testDto.setOther(other);List<TestDto.Other> list = new ArrayList<>();for (int i = 0; i <= 2; i++) {TestDto.Other other1 = new TestDto.Other();other1.setCareer("码农" + i);other1.setCompany("字节不跳动" + i);list.add(other1);}testDto.setOtherList(list);PdfUtil.generatePdf(request, response, "test.pdf", testDto, false);}}

浏览器访问ip:port/test/pdf,其中ip为你的ip地址,port为你的端口,访问结果如下:

在这里插入图片描述

这篇关于java集成itextpdf实现通过pdf模板填充数据生成pdf的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/165017

相关文章

Java Stream流与使用操作指南

《JavaStream流与使用操作指南》Stream不是数据结构,而是一种高级的数据处理工具,允许你以声明式的方式处理数据集合,类似于SQL语句操作数据库,本文给大家介绍JavaStream流与使用... 目录一、什么是stream流二、创建stream流1.单列集合创建stream流2.双列集合创建str

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

PHP轻松处理千万行数据的方法详解

《PHP轻松处理千万行数据的方法详解》说到处理大数据集,PHP通常不是第一个想到的语言,但如果你曾经需要处理数百万行数据而不让服务器崩溃或内存耗尽,你就会知道PHP用对了工具有多强大,下面小编就... 目录问题的本质php 中的数据流处理:为什么必不可少生成器:内存高效的迭代方式流量控制:避免系统过载一次性

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv