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

相关文章

Python在二进制文件中进行数据搜索的实战指南

《Python在二进制文件中进行数据搜索的实战指南》在二进制文件中搜索特定数据是编程中常见的任务,尤其在日志分析、程序调试和二进制数据处理中尤为重要,下面我们就来看看如何使用Python实现这一功能吧... 目录简介1. 二进制文件搜索概述2. python二进制模式文件读取(rb)2.1 二进制模式与文本

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

Java中Map的五种遍历方式实现与对比

《Java中Map的五种遍历方式实现与对比》其实Map遍历藏着多种玩法,有的优雅简洁,有的性能拉满,今天咱们盘一盘这些进阶偏基础的遍历方式,告别重复又臃肿的代码,感兴趣的小伙伴可以了解下... 目录一、先搞懂:Map遍历的核心目标二、几种遍历方式的对比1. 传统EntrySet遍历(最通用)2. Lambd

Spring Boot 中 RestTemplate 的核心用法指南

《SpringBoot中RestTemplate的核心用法指南》本文详细介绍了RestTemplate的使用,包括基础用法、进阶配置技巧、实战案例以及最佳实践建议,通过一个腾讯地图路线规划的案... 目录一、环境准备二、基础用法全解析1. GET 请求的三种姿势2. POST 请求深度实践三、进阶配置技巧1

springboot+redis实现订单过期(超时取消)功能的方法详解

《springboot+redis实现订单过期(超时取消)功能的方法详解》在SpringBoot中使用Redis实现订单过期(超时取消)功能,有多种成熟方案,本文为大家整理了几个详细方法,文中的示例代... 目录一、Redis键过期回调方案(推荐)1. 配置Redis监听器2. 监听键过期事件3. Redi

Spring Boot 处理带文件表单的方式汇总

《SpringBoot处理带文件表单的方式汇总》本文详细介绍了六种处理文件上传的方式,包括@RequestParam、@RequestPart、@ModelAttribute、@ModelAttr... 目录方式 1:@RequestParam接收文件后端代码前端代码特点方式 2:@RequestPart接

SpringBoot整合Zuul全过程

《SpringBoot整合Zuul全过程》Zuul网关是微服务架构中的重要组件,具备统一入口、鉴权校验、动态路由等功能,它通过配置文件进行灵活的路由和过滤器设置,支持Hystrix进行容错处理,还提供... 目录Zuul网关的作用Zuul网关的应用1、网关访问方式2、网关依赖注入3、网关启动器4、网关全局变

SpringBoot全局异常拦截与自定义错误页面实现过程解读

《SpringBoot全局异常拦截与自定义错误页面实现过程解读》本文介绍了SpringBoot中全局异常拦截与自定义错误页面的实现方法,包括异常的分类、SpringBoot默认异常处理机制、全局异常拦... 目录一、引言二、Spring Boot异常处理基础2.1 异常的分类2.2 Spring Boot默

基于SpringBoot实现分布式锁的三种方法

《基于SpringBoot实现分布式锁的三种方法》这篇文章主要为大家详细介绍了基于SpringBoot实现分布式锁的三种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、基于Redis原生命令实现分布式锁1. 基础版Redis分布式锁2. 可重入锁实现二、使用Redisso

SpringBoo WebFlux+MongoDB实现非阻塞API过程

《SpringBooWebFlux+MongoDB实现非阻塞API过程》本文介绍了如何使用SpringBootWebFlux和MongoDB实现非阻塞API,通过响应式编程提高系统的吞吐量和响应性能... 目录一、引言二、响应式编程基础2.1 响应式编程概念2.2 响应式编程的优势2.3 响应式编程相关技术