Java使用pdf模板生成pdf文件_java根据pdf模板生成pdf-程序员宅基地

技术标签: java  Java日记  

前言

有时候,业务中我们需要使用pdf模板生成一份pdf文件。如何生成pdf文件?
如下,我们把左侧的模板生成为右侧的填充文件。

在这里插入图片描述

如何生成pdf文件

1. 制作pdf模板

(1)首先创建一份template.docx文件,另存为template.pdf
在这里插入图片描述
(2) 打开pdf阅读器,点击 创建>> 表单 >> 从文件,选择刚刚的template.pdf文件

这里使用的福昕PDF编辑器阅读器,其它能编辑pdf的软件亦可(Adobe Acrobat DC)

在这里插入图片描述
(3) 选择表单>>文本域,双击待填充的位置,填写名称,最后保存得到pdf的模板。

在这里插入图片描述
在这里插入图片描述

2. 填充pdf模板生成文件

  1. 导入依赖
<dependency>
            <groupId>com.itextpdf</groupId>
            <artifactId>itextpdf</artifactId>
            <version>5.5.13.3</version>
        </dependency>
  1. 测试用例填充word模板
package com.it2.pdfdemo;

import com.itextpdf.text.BadElementException;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.Image;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.*;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * 使用pdf模板生成pdf文件
 */
@SpringBootTest
@Slf4j
class PdfdemoApplicationTests {
    

    @Test
    void testPdf() {
    
        String inputFileName = "D:\\test\\template6.pdf";//模板
        String outputFileName = "D:\\test\\template6_out.pdf";//生成文件

        // 填充表单的数据(文本)
        Map<String, Object> data = new HashMap<String, Object>();
        data.put("name", "王多余");
        data.put("birthday", "2001-12-23");
        data.put("sex", "男");
        data.put("address", "北京市朝阳区和平路101号");
        data.put("phone", "13600000000");

        // 填充表单的数据(图片)
        Map<String, String> picData = new HashMap<>();
        picData.put("pic", "D:\\test\\dog.png");

        OutputStream os = null;
        PdfStamper ps = null;
        PdfReader reader = null;
        try {
    
            // 1 获取文件的输出流
            os = new FileOutputStream(new File(outputFileName));
            // 2 读取pdf模板
            reader = new PdfReader(inputFileName);
            // 3 根据表单生成一个新的pdf
            ps = new PdfStamper(reader, os);
            // 4 获取pdf表单
            AcroFields form = ps.getAcroFields();
            // 5 给表单添加中文字体
            BaseFont bf = BaseFont.createFont("Font/SIMYOU.TTF", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
            form.addSubstitutionFont(bf);

            // 6遍历data赋值到form
            for (String key : data.keySet()) {
    
                form.setField(key, data.get(key).toString());
            }
            ps.setFormFlattening(true);

            // 7 填充图片到form
            PdfStamper stamper = ps;
            picData.forEach((fieldName, imgSrc) -> {
    
                try {
    
                    List<AcroFields.FieldPosition> fieldPositions = form.getFieldPositions(fieldName);
                    for (AcroFields.FieldPosition fieldPosition : fieldPositions) {
    
                        // 通过域名获取所在页和坐标,左下角为起点
                        int pageno = fieldPosition.page;
                        Rectangle signrect = fieldPosition.position;
                        float x = signrect.getLeft();
                        float width = signrect.getWidth();
                        float y = signrect.getBottom();
                        // 读图片
                        Image image = Image.getInstance(imgSrc);
                        float img_width = image.getWidth();
                        // 获取操作的页面
                        PdfContentByte under = stamper.getOverContent(pageno);
                        // 图片的大小
                        image.scaleToFit(signrect.getWidth(), signrect.getHeight());
                        // 添加图片
                        image.setAbsolutePosition(x, y);
                        under.addImage(image);
                    }
                } catch (BadElementException e) {
    
                    e.printStackTrace();
                } catch (IOException e) {
    
                    e.printStackTrace();
                } catch (DocumentException e) {
    
                    e.printStackTrace();
                }
            });

            log.info("生成pdf成功:{}", outputFileName);
        } catch (Exception e) {
    
            log.error("生成pdf失败:{}", outputFileName);
            e.printStackTrace();
        } finally {
    
            try {
    
                ps.close();
                reader.close();
                os.close();
            } catch (Exception e) {
    
                e.printStackTrace();
            }
        }
    }

}
  1. 引入字体文件(示例为 SIMYOU.TTF),存在位置resources\Font目录下,demo使用的字体文件为简体幼圆常规

提示:由于部分字体存在可能的商业侵权问题,如您在商业使用,建议使用非侵权字体,或者事先已获取所使用的字体授权

在这里插入图片描述
在这里插入图片描述

  1. 运行测试用例结果
    在这里插入图片描述

在这里插入图片描述

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u011628753/article/details/131377253

智能推荐

IDW Spatial Analyst(ESRI.ArcGIS.SpatialAnalystTools)ArcEngine中反距离加权插值方法_c# arcengine 反距离权重插值-程序员宅基地

文章浏览阅读4k次。IDW(反距离加权插值方法)public void GPIDW(AxMapControl mapControl, ILayer layer, IEnvelope e, string pField, bool ISUPPER, string tucengname, double rasterPixel, double dInterval, ISpatialReference spatialR_c# arcengine 反距离权重插值

QT基础练习-程序员宅基地

文章浏览阅读798次,点赞18次,收藏29次。槽函数。

在Windows平台编译OpenCV及开发环境配置_opencv 编译 环境变量-程序员宅基地

文章浏览阅读1.5k次,点赞3次,收藏9次。下载失败后,文件夹中会出现文件:download_with_curl.sh,用文本编辑器或者直接双击打开,在所有链接前面添加https://ghproxy.com/,复制命令到终端执行,文件里每个下载失败的文件都要执行一次,如图八。在编译过程中,编译器会将源代码转换为目标代码(.o文件),然后链接器将目标代码和其他依赖项(如库文件)合并为可执行文件或动态链接库(.dll、.so文件),以供应用程序调用和使用。遇到的问题,不一定完全一致,如果出现的问题如上述三点,或比较相似,以下是提供的建议。_opencv 编译 环境变量

Java——File类、递归_java递归创建文件-程序员宅基地

文章浏览阅读138次。Java——File类、递归_java递归创建文件

吴恩达深度学习笔记整理——RMSprop算法_吴恩达 rmsprop-程序员宅基地

文章浏览阅读92次。为了确保√sdw和√sdb不为0,则训练过程将sdw+E(epsilon),E(epsilon)是一个非常小的非0的正数,例如:10……这样当db变化越大,sdb也会变大,那么由于计算b时,需要除以√Sdb,进而整体数值越小,那么缩小了b的震荡幅度,w也是类似的。通过对梯度进行加权平均来计算梯度的变化率。这个加权平均称为“动量”,它可以帮助RMSprop在。利用加权平均值调整了权重值和偏差值的震荡幅度,进而提高了训练的稳定性和速度。迭代进行Mini-batch,针对每次迭代t:。来控制动量的更新速度。_吴恩达 rmsprop

抓rtmp推流地址_在浏览器中实现RTMP推流-程序员宅基地

文章浏览阅读2.6k次。什么是RTMPRTMP(Real-Time Messaging Protocol) 是为了能在基于Adobe Flash平台的技术间实现音频、视频及数据的高性能传输而设计的,包括Adobe Flash Player和Adobe AIR。现在,RTMP已经可以被作为一项开放的标准,用于那些支持视频、音频及其他数据传输的产品或技术,只要传输的数据格式能与Adobe Flash Player适配(如sw..._浏览器截推流

随便推点

Anaconda安装教程傻瓜教程_anaconda下没有usr目录-程序员宅基地

文章浏览阅读4.7w次,点赞119次,收藏717次。网上好多教程找的我心好累,为了方便新手安装,我这边整理了大致详细的教程,参考了好几个大佬的教程,废话不多说,直接开始~**前言:**有人纠结先安装python还是安装anaconda,这边的建议是装anaconda,就不需要单独装python了。其次,有小伙伴的Anaconda的安装分两种情况:情况一:电脑现在没有装python或者现在装的可以卸载掉(装Anaconda时先卸python);情况二:电脑目前装了python,但想保留它;_anaconda下没有usr目录

防止多次触发click事件_@click="unbind-程序员宅基地

文章浏览阅读1.2k次。一、问题描述 用户频繁触发同一表单的提交按钮可能会导致流程或程序紊乱二、需求 满足问题需求,在不妨碍用户使用的情况下禁止程序多次被触发三、实现逻辑 完成触发后解绑点击/相关事件四、实现代码1、知识点补充unbind()方法 (1) 官方说明:unbind() 方法移除被选元素的事件处理程序。该方法能够移除所有的或被选的事件处理程序,或者当事件发生时终止指定函数的运行。 (2)简单理解:解绑,与绑定相对2、代码<%..._@click="unbind

谷歌账号风控:个人开发者账号的问题分析与应对策略_issue found: high risk behavior googlr-程序员宅基地

文章浏览阅读699次,点赞10次,收藏7次。去年9-10月,众多开发者目睹了谷歌大规模封禁个人账号的举动,并不断更新了风控政策。这是因为随着个人开发者账号滥用行为的增加,谷歌不得不采取更为强硬的措施来保护其平台的良好环境。那目前谷歌账号风控措施_issue found: high risk behavior googlr

面试官:localhost 和 127.0.0.1 的区别是什么?问倒一大片。。。-程序员宅基地

文章浏览阅读20次。戳下方名片,关注并星标!回复“1024”获取2TB学习资源!????体系化学习:运维工程师打怪升级进阶之路 4.0— 特色专栏—MySQL/PostgreSQL/MongoDBElasticSearch/Hadoop/RedisKubernetes/Docker/DevOpsKafka/RabbitMQ/Zookeeper监控平台/应用与服务/集群管理N...

CVPR 2022 | Glow模型助力黑盒对抗攻击-程序员宅基地

文章浏览阅读514次。近日,清华大学,香港中文大学(深圳)等单位联合发表了一篇在对抗机器学习领域中黑盒攻击场景下的论文,已顺利被CVPR 2022接收。通过迁移一部分替代模型的条件对抗分布(CAD)的参数,同时根据对目标模型的查询学习剩下未迁移的参数,所提出的方法可以在任何新的正常样本上调整目标模型的 CAD以提高攻击性能。论文标题: Boosting Black-Box Attack with Partially T..._流模型 黑盒攻击

mock.js的基本使用_mockjs url匹配规则-程序员宅基地

文章浏览阅读1.1k次。mock模拟数据_mockjs url匹配规则