IText创建加盖公章的pdf文件并生成压缩文件

第一、前言
此前已在文章:Java使用IText根据pdf模板创建pdf文件介绍了Itex的基本使用技巧,本篇以一个案例为基础,主要介绍IText根据pdf模板填充生成pdf文件,并生成压缩文件。

第二、案例
以下面pdf模板为例,生成一个pdf文件,并将其压缩成zip文件。
在这里插入图片描述
在这里插入图片描述
第三、代码实现
1、将pdf模板机公章图片放到resource目录;
在这里插入图片描述
2、为提高效率,在系统启动后将模板加载到redis缓存中,代码如下;

import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.util.Objects;
import javax.annotation.PostConstruct;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPoolAbstract;
@Component
public class InitConfig {private static final Logger logger = LoggerFactory.getLogger(InitConfig.class);private final JedisPoolAbstract jedisPool;@Autowiredpublic InitConfig(JedisPoolAbstract jedisPool) {super();this.jedisPool = jedisPool;}@PostConstructpublic void init() {try {Jedis jedis = jedisPool.getResource();final byte[] key="data".getBytes();if (jedis.exists(key)) {jedis.del(key);}InputStream is = this.getClass().getResourceAsStream("/template/data.pdf");jedis.set(key, streamToByteArray(is));final byte[] gongzhangKey="gongzhang".getBytes();if (jedis.exists(gongzhangKey)) {jedis.del(gongzhangKey);}is = this.getClass().getResourceAsStream("/image/gongzhang.png");jedis.set(gongzhangKey, streamToByteArray(is));} catch (Exception e) {logger.error("加载模板异常", e);}}public static byte[] streamToByteArray(InputStream is) throws Exception {ByteArrayOutputStream bos = null;try {bos = new ByteArrayOutputStream();byte[] b = new byte[1024];int len;while ((len = is.read(b)) != -1) {bos.write(b, 0, len);}return bos.toByteArray();} catch (Exception e) {logger.error("初始化模板", e);throw e;} finally {if (Objects.nonNull(bos)) {bos.close();}}}
}

3、操作文件工具类;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.Map;
import java.util.Objects;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
import com.itextpdf.text.Image;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.AcroFields;
import com.itextpdf.text.pdf.BaseFont;
import com.itextpdf.text.pdf.PdfContentByte;
import com.itextpdf.text.pdf.PdfReader;
import com.itextpdf.text.pdf.PdfStamper;
import com.itextpdf.text.pdf.PdfWriter;public final class CommonFileUtil {private final static org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(CommonFileUtil.class);/*** 根据模板生成文件* @param paramMap* @param deskFile* @param template* @param gongZhang* @throws Exception*/public static void create(Map<String, Object> paramMap, File deskFile, final byte[] template, final byte[] gongZhang) throws Exception {PdfReader reader = null;PdfStamper stamp = null;try {reader = new PdfReader(template);stamp = new PdfStamper(reader, new FileOutputStream(deskFile));stamp.setEncryption(null, "lsy2024".getBytes(), PdfWriter.ALLOW_PRINTING, true);//取出报表模板中的所有字段AcroFields form = stamp.getAcroFields();//设置宋体BaseFont song = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);//设置参数for (Map.Entry<String, Object> entry : paramMap.entrySet()) {String key = entry.getKey();form.setFieldProperty(key, "textfont", song, null);form.setField(key, entry.getValue().toString());}//插入公章CommonFileUtil.insertImage(form, stamp, "gongzhang", gongZhang);//保存修改stamp.setFormFlattening(true);} catch (Throwable e) {logger.error("deskFile:{};文件生成失败!", deskFile, e);throw e;} finally {if (Objects.nonNull(stamp)) {stamp.close();}if (Objects.nonNull(reader)) {reader.close();}}}/*** pdf模板插入图片* @param form* @param stamper* @param filedName* @param gongZhang* @throws Exception*/public static void insertImage(AcroFields form, PdfStamper stamper, String filedName, final byte[] gongZhang) throws Exception {final Rectangle signRect = form.getFieldPositions(filedName).get(0).position;float x = signRect.getLeft();float y = signRect.getBottom();Image image = Image.getInstance(gongZhang);// 获取操作的页面PdfContentByte under = stamper.getOverContent(form.getFieldPositions(filedName).get(0).page);// 根据域的大小缩放图片image.scaleToFit(signRect.getWidth(), signRect.getHeight());// 添加图片image.setAbsolutePosition(x, y);under.addImage(image);}/*** 文件压缩* @param sourcePath* @param zipFilePath* @throws Exception*/public static void encryptNoPassword(String sourcePath, String zipFilePath) throws Exception {final long start = System.currentTimeMillis();byte[] buf = new byte[1024];File zipFile = new File(zipFilePath);//zip文件不存在,则创建文件,用于压缩ZipOutputStream zos = null;try {if (!zipFile.exists()) {zipFile.createNewFile();}zos = new ZipOutputStream(new FileOutputStream(zipFile));File file = new File(sourcePath);for (File sourceFile : file.listFiles()) {if (sourceFile == null || !sourceFile.exists()) {continue;}try (FileInputStream fis = new FileInputStream(sourceFile)) {//直接放到压缩包的根目录zos.putNextEntry(new ZipEntry(sourceFile.getName()));int len;while ((len = fis.read(buf)) > 0) {zos.write(buf, 0, len);}zos.closeEntry();}}} catch (Throwable e) {logger.error("sourcePath:{};zipFilePath:{};压缩文件失败!", sourcePath, zipFilePath, e);throw e;} finally {if (zos != null) {zos.close();}}logger.info("sourcePath:{};zipFilePath:{};压缩文件结束!{}", sourcePath, zipFilePath, System.currentTimeMillis() - start);}/*** 清理文件* @param path*/public static void clean(String path) {File rootPath = new File(path);if (rootPath.exists()) {if (rootPath.isDirectory()) {for (File file : rootPath.listFiles()) {clean(file.getPath());}}if (rootPath.isFile() || rootPath.listFiles().length == 0) {rootPath.delete();}}}
}

4、编写一个测试方法;

    public void print() {try {final String name = "张三";final String rootPath = "D:/temp/";File rootFile = new File(rootPath);if (!rootFile.exists()) {rootFile.mkdir();}File destFile = new File(rootPath + name + ".pdf");Map<String, Object> paramMap = new HashMap<String, Object>();paramMap.put("name", name);paramMap.put("birthDate", "2000-01-01");paramMap.put("currentDate", "2024  年  11  月  22  日");CommonFileUtil.create(paramMap, destFile, this.jedisPool.getResource().get("data".getBytes()), this.jedisPool.getResource().get("gongzhang".getBytes()));CommonFileUtil.encryptNoPassword(rootPath, "D:/data.zip");} catch (Exception e) {logger.error("==" + e);}}

5、调用并执行测试方法,将会在D盘创建压缩文件;
在这里插入图片描述
6、解压后如下;
在这里插入图片描述

欢迎大家积极留言交流学习心得!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/476760.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

组会 | 大语言模型 + LoRA

目录 1 大语言模型概述1.1 模型的架构1.2 模型的细节&#xff1a;标记化和嵌入化1.3 模型的核心 2 多头注意力机制3 LoRA 概述3.1 冻结部分模型参数3.2 低秩适配&#xff08;LoRA&#xff09;3.2.1 核心工作原理&#xff1a;冻结模型参数3.2.2 核心工作原理&#xff…

对象:是什么,使用,遍历对象,内置对象

对象使用&#xff1a; 对象访问&#xff1a;&#xff08;对象每个属性之间用逗号隔开&#xff09; 补充&#xff1a;也可以通过 对象名[‘属性名’] 对象方法&#xff1a; 方法名:匿名函数 调用方法不需要控制台打印&#xff0c;只要调用就自动输出值 遍历对象&#xff1a; …

小程序24-滚动效果:scroll-view组件详解

在微信小程序中如果想实现内容滚动&#xff0c;需要使用 scroll-view 组件 scroll-view&#xff1a;可滚动视图区域&#xff0c;适用于需要滚动展示内容的场景&#xff0c;用户可以通过手指滑动或者点击滚动条滚动内容。 scroll-x允许横向滚动scroll-y允许纵向滚动 实现横向…

C++设计模式行为模式———中介者模式

文章目录 一、引言二、中介者模式三、总结 一、引言 中介者模式是一种行为设计模式&#xff0c; 能让你减少对象之间混乱无序的依赖关系。 该模式会限制对象之间的直接交互&#xff0c; 迫使它们通过一个中介者对象进行合作。 中介者模式可以减少对象之间混乱无序的依赖关系&…

AUTOSAR_EXP_ARAComAPI的7章笔记(6)

☞返回总目录 相关总结&#xff1a;ara::com 与 AUTOSAR 元模型的关系总结 7.4 ara::com 与 AUTOSAR 元模型的关系 在本文档中&#xff0c;我们一直在不涉及具体的AP元模型&#xff08;其清单部分&#xff09;的情况下解释 ara::com API的思想和机制&#xff0c;AP元模型是正…

LINUX系统编程之——环境变量

目录 环境变量 1、基本概念 2、查看环境变量的方法 三、查看PATH环境变量的內容 1&#xff09;不带路径也能运行的自己的程序 a、将自己的程序直接添加到PATH指定的路径下 b、将程序所在的路径添加到PATH环境中 四、环境变量与本地变量 1、本地变量创建 2、环境变量创…

MacOS通过X11转发远程运行virt-manager进行虚机分配

今天需要通过本地macbook机器连接远程物理机&#xff0c;执行虚机分配&#xff0c;现有文档仅提供window环境安装&#xff0c;如下整理Mac环境下的安装步骤 操作篇 前提条件 支持x11转发的terminal&#xff0c;我本地使用iTerm2&#xff1b;本地安装XQuartz&#xff0c;作为…

【AI系统】AI 基本理论奠定

虽然 AI 在今年取得了举世瞩目的进展与突破&#xff0c;但是其当前基于的核心理论神经网络等&#xff0c;在这波浪潮开始前已经基本奠定&#xff0c;并经历了多次的起起伏伏。神经网络作为 AI 的前身&#xff0c;经历了以下的发展阶段&#xff1a; 萌芽兴奋期&#xff08;约 19…

网络安全服务人才发展路线图

到2023年&#xff0c;全球网络安全支出规模将达到1512亿美元&#xff08;约合10640.4亿元人民币&#xff09;&#xff0c;并将以9.4%的年复合增长率持续增长。与火爆的产业现状相比&#xff0c;中国的网络安全服务人才面临巨大缺口。相关数据显示&#xff0c;我国网络安全人才缺…

STM32 ADC 读取模拟量

问题 我有一个调速开关&#xff0c;模拟量输入&#xff0c;因此需要使用 STM32 读取模拟量&#xff0c;并通过串口输入来调试。串口相关知识参考 STM32 串口输出调试信息。 硬件信息: CubeMX version 6.12.1Keil uVision V5.41.0.0 参考知识 【STM32】HAL库 STM32CubeMX教…

[每周一更]-(第124期):模拟面试|缓存面试思路解析

文章目录 31 为什么 Redis 不立刻删除已经过期的数据?1. Redis 是怎么删除过期 key 的?2. Redis 为什么不立刻删除已经过期的 key?3. Redis 为什么不每个 key 都启动一个定时器,监控过期时间?4. Redis 是如何执行定期删除的?5. 为什么 Redis 在定期删除的时候不一次性把所…

操作系统——揭开盖子

计算机执行时——取指执行 es:bx等于从0x9000开始&#xff0c;到0x90200结束

uni-app 认识条件编译,了解多端部署

一. 前言 在使用 uni-app 进行跨平台开发的过程中&#xff0c;经常会遇到需要针对不同平台或不同环境进行条件编译的情况。条件编译是一种在编译过程中根据指定条件选择不同代码路径的技术&#xff0c;可以帮助我们在不同平台或环境下编写不同的代码&#xff0c;以适应不同的平…

模糊控制系统的设计(取材bilibili_蓝天的季洁)

模糊控制原理和传统控制原理&#xff0c;在框图上的区别实际上只在控制器方面存在差异&#xff0c;将传统的控制器改为了模糊控制器&#xff08;fuzzy controller&#xff09;。 通过举例说明&#xff0c;将原有的[0,100]的参数通过隶属函数规则&#xff0c;&#xff08;类似于…

pytest日志总结

pytest日志分为两类&#xff1a; 一、终端&#xff08;控制台&#xff09;打印的日志 1、指定-s&#xff0c;脚本中print打印出的信息会显示在终端&#xff1b; 2、pytest打印的summary信息&#xff0c;这部分是pytest 的默认输出&#xff08;例如测试结果PASSED, FAILED, S…

向量数据库FAISS之四:向量检索和 FAISS

来自 YouTube 1.相似度搜索的传统方法(Jaccard, w-shingling, Levenshtein) 1.Jaccard 距离 公式 Jaccard ( A , B ) 1 − ∣ A ∩ B ∣ ∣ A ∪ B ∣ \text{Jaccard}(A, B) 1 - \frac{|A \cap B|}{|A \cup B|} Jaccard(A,B)1−∣A∪B∣∣A∩B∣​ 其中&#xff0c; A 和 …

深入探究蓝牙节能技术:SNIFF与HOLD模式

目录 一、概述 1.1. Sniff Mode&#xff08;嗅探模式/呼吸模式&#xff09; 1.1.1.定义与目的 1.1.2 工作原理 1.1.3 进入与退出 1.2. Hold Mode&#xff08;保持模式&#xff09; 1.2.1. 定义与目的 1.2.2. 工作原理 1.2.3. 进入 1.2.4. 通知机制 二、Sniff mode&a…

Linux驱动开发快速入门——字符设备驱动(直接操作寄存器设备树版)

Linux驱动开发快速入门——字符设备驱动 前言 笔者使用开发板型号&#xff1a;正点原子的IMX6ULL-alpha开发板。ubuntu版本为&#xff1a;20.04。写此文也是以备忘为目的。 字符设备驱动 本小结将以直接操作寄存器的方式控制一个LED灯&#xff0c;可以通过read系统调用可以…

ROS机器视觉入门:从基础到人脸识别与目标检测

前言 从本文开始&#xff0c;我们将开始学习ROS机器视觉处理&#xff0c;刚开始先学习一部分外围的知识&#xff0c;为后续的人脸识别、目标跟踪和YOLOV5目标检测做准备工作。我采用的笔记本是联想拯救者游戏本&#xff0c;系统采用Ubuntu20.04&#xff0c;ROS采用noetic。 颜…

电子电气架构 ---漫谈车载网关

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 所有人的看法和评价都是暂时的,只有自己的经历是伴随一生的,几乎所有的担忧和畏惧,都是来源于自己的想象,只有你真的去做了,才会发现有多快乐。…