java导出pdf文件

java导出pdf,前端下载

    • 1、制作pdf模板
    • 2、获取pdf导出中文需要的文件
    • 3、实现
    • 4、前端发起请求并生成下载链接

使用注意点
因为原来制作的pdf表单内容过于复杂,下面代码只包含前两行的操作。
本次操作需要前端向后端发起请求,后端返回数据给前端用于下载,所以没有在本地进行保存。
第 1 步制作pdf模板需要的pdf编辑软件基本上都需要钱,可以去买一个
第 2 步获取的pdf导出的中文需要的文件,如果pdf输出的内容有中文就需要去弄一下这个文件,在代码中用于读取设置中文字体

包含内容
1、导出pdf
2、设置斜体水印

1、制作pdf模板

先将需要的pdf模板通过word制作出来,然后导出为pdf
先准备一个pdf
使用Adobe Acrobat DC 打开并制作模板(其他pdf编辑软件也可以)
在这里插入图片描述
选择打开前面导出的pdf模板
在这里插入图片描述
点击准备表单
在这里插入图片描述
点击之后,可以针对没一个位置进行编辑,选中双击就可以进行编辑了,要注意,每个位置的名字都需要是唯一的
在这里插入图片描述
全部赋值后保存即可

2、获取pdf导出中文需要的文件

获取中文字体需要的文件
在这里插入图片描述
在电脑这个路径下选择下载一个就行
在这里插入图片描述

3、实现

pom依赖

<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itext7-core</artifactId>
    <version>7.2.3</version>
    <type>pom</type>
</dependency>

<dependency>
    <groupId>org.apache.pdfbox</groupId>
    <artifactId>pdfbox</artifactId>
    <version>2.0.24</version>
</dependency>

controller接口

    @GetMapping("/exportPDF/{applyId}")
    public ResponseEntity<byte[]> exportPDF(@PathVariable("applyId") String applyId, HttpServletResponse response) throws IOException,ParseException,Exception {

        byte[] res = applyService.exportPDF(applyId);
        HttpHeaders headers = new HttpHeaders();
        headers.add("Content-Disposition", "attachment; filename=filled_form.pdf");
        headers.add("Content-Type", "application/pdf");

        return ResponseEntity.ok()
                .headers(headers)
                .body(res);

    }

service具体实现

	public static byte[] exportPDF1() throws Exception {
        String inputTemplateName = "template";
        try {
            pdfBytes = null;
            Map<String, String> map = new HashMap<>();
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");

            // map预填数据,用于后面读取输出到pdf文件上
            map.put("department-1", "研发中心");
            map.put("submitDate-1", sdf.format(new Date()));
            map.put("submitPerson-1", "张三");
            map.put("travelPerson-1", "李四");
            map.put("receivePerson-1","王五");

            // 设置中文字体
            PdfFont chineseFont = PdfFontFactory.createFont("src/main/resources/file/simsun.ttc,0");

            // 模板路径
            String templatePath = "src\\main\\resources\\file\\" + inputTemplateName + ".pdf";

            // 重点,这一个关联了reader 和 writer
            ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();

            // 读取文件
            FileInputStream pdfInputStream = new FileInputStream(new File(templatePath));
            // 定义 reader 和writer
            PdfReader reader = new PdfReader(pdfInputStream);
            PdfWriter writer = new PdfWriter(byteArrayOutputStream,new WriterProperties().setStandardEncryption(
                    null,
                    null,
                    EncryptionConstants.ALLOW_PRINTING, // 允许打印
                    EncryptionConstants.ENCRYPTION_AES_128 // 加密方式
            ));

            // 根据 reader 和writer 创建 PdfDocument
            PdfDocument pdfDocument = new PdfDocument(reader,writer);

            // 下面是给文件添加水印,不需要的可以直接删掉
            // 获取 pdf 模板页数
            int numberOfPages = pdfDocument.getNumberOfPages();

            // 遍历每一页并添加水印
            for (int i = 1; i <= numberOfPages; i++) {
                PdfPage page = pdfDocument.getPage(i);

                // 获取页面尺寸(在这里我没有用)
                int pageWidth = (int) Math.floor(page.getPageSize().getWidth());
                int pageHeight = (int) Math.floor(page.getPageSize().getHeight());

                // 创建一个 PdfCanvas 对象
                PdfCanvas canvas = new PdfCanvas(page);

                // 保存当前坐标系状态
                canvas.saveState();

                // 水印内容旋转
                double angle = Math.toRadians(45);
                double cos = Math.cos(angle);
                double sin = Math.sin(angle);
                canvas.concatMatrix(cos, sin, -sin, cos, 0, 0);

                // 设置水印的字体和透明度
                canvas.setFontAndSize(PdfFontFactory.createFont(), 20);
                canvas.setFillColorRgb(0.75f, 0.75f, 0.75f);  // 灰色
                canvas.setLineWidth(2);

                // 正常应该根据获取到的页面尺寸进行 x y 轴的遍历并
                // 但是我这边没有铺满,就自己设置了遍历的范围
                for (int x = -90; x < 2000; x += 300) {
                    for (int y = -190; y < 2000; y += 200) {
                        // 绘制水印文字
                        canvas.beginText();
                        canvas.setTextMatrix(x, y);  // 设置水印位置
                        canvas.showText("Watermark Text this is just a test");  // 水印文字内容
                        canvas.endText();
                    }
                }

                // 恢复坐标系状态
                canvas.restoreState();

            }

            // form 可以理解为把pdf文件看做一个form表单,以key value键值对保存
            PdfAcroForm form = PdfAcroForm.getAcroForm(pdfDocument, true);

            // 遍历上面预填的 map 并将值根据 key 赋值到form
            for (Map.Entry<String, String> entry : map.entrySet()) {
                form.getField(entry.getKey()).setValue(entry.getValue());
                form.getField(entry.getKey()).setFont(chineseFont).setFontSize(8);
            }
            pdfDocument.close();
            // 返回文件流
            pdfBytes = byteArrayOutputStream.toByteArray();

            return pdfBytes;

        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            return pdfBytes;
        }
    }

4、前端发起请求并生成下载链接

	exportPdf(applyId) {
      exportPDF(applyId).then(res => {
        // 创建一个 Blob 对象,指定类型为 PDF 文件
        const blob = new Blob([res.data], { type: 'application/pdf' });

        // 创建一个 URL 对象,指向 Blob 数据
        const url = URL.createObjectURL(blob);

        // 创建一个下载链接
        const link = document.createElement('a');
        link.href = url;
        link.download = 'generated_with_form.pdf'; // 设置下载文件名

        // 模拟点击下载链接
        link.click();

        // 下载完成后释放 URL 对象
        URL.revokeObjectURL(url);
      })
    },

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

1月13日学习

[HITCON 2017]SSRFme 直接给了源代码&#xff0c;题目名称还是ssrf&#xff0c;那么该题大概率就是SSRF的漏洞&#xff0c;进行代码审计。 <?php// 检查是否存在 HTTP_X_FORWARDED_FOR 头&#xff0c;如果存在&#xff0c;则将其拆分为数组&#xff0c;并将第一个 IP 地址…

在一个sql select中作多个sum并分组

有表如下&#xff1b; 单独的对某一个列作sum并分组&#xff0c;结果如下&#xff1b; 对于表的第7、8行&#xff0c;num1都有值&#xff0c;num2都是null&#xff0c;对num2列作sum、按id分组&#xff0c;结果在id为4的行会显示一个null&#xff1b; 同时对2个列作sum&#x…

[Deep Learning] Anaconda+CUDA+CuDNN+Pytorch(GPU)环境配置-2025

文章目录 [Deep Learning] AnacondaCUDACuDNNPytorch(GPU)环境配置-20250. 引子1. 安装Anaconda1.1 安装包下载&#xff1a;1.2 启用安装包安装1.3 配置(系统)环境变量1.4 验证Anaconda是否安装完毕1.5 Anaconda换源 2. 安装CUDACuDNN2.1 判断本机的CUDA版本2.2 下载适合自己CU…

不需要配置文件实现Javaweb项目的启动

1.首先看一下web.xml主要配置内容 <?xml version"1.0" encoding"UTF-8"?> <web-app xmlns"http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://xm…

【网络篇】IP知识

IPv4首部与IPv6首部 IPv4相对于IPv6的好处&#xff1a; 1.IPv6可自动配置&#xff0c;即使没有DHCP服务器也可以实现自动分配IP地址&#xff0c;实现即插即用。 2.IPv6包首部长度采用固定40字节&#xff0c;删除了选项字段&#xff0c;以及首部校验和&#xff0c;简化了首部…

MyBatis核心流程

目录 数据处理的发展 MyBatis概述 ​编辑 MyBatis核心流程 观察测试类 重要对象和流程 SqlSessionFactory [初始化] 创建SqlSession会话对象 创建XxxMapper[代理]对象 执行SQL操作 [复杂一丢丢] ​编辑 数据处理的发展 1.原生JDBC 2. DBUtils工具类 [jdbctemp..] 3. …

低代码独特架构带来的编译难点及多线程解决方案

前言 在当今软件开发领域&#xff0c;低代码平台以其快速构建应用的能力&#xff0c;吸引了众多开发者与企业的目光。然而&#xff0c;低代码平台独特的架构在带来便捷的同时&#xff0c;也给编译过程带来了一系列棘手的难点。 一&#xff0c;低代码编译的难点 &#xff08;1…

Spring Security单点登录

本文介绍了Spring Security单点登录的概念和基本原理。单点登录是指用户只需登录一次&#xff0c;即可在多个相互信任的系统中实现无缝访问和授权。通过Spring Security框架的支持&#xff0c;可以实现有效的用户管理和权限控制。最后&#xff0c;本文提供了实际应用案例&#…

查找某个年龄段的用户信息TCP头格式为什么需要 TCP 协议? TCP 工作在哪一层?

查找某个年龄段的用户信息 select device_id,gender,age from user_profile where age>20 and age<23; TCP头格式 序列号&#xff1a;在建立连接时由计算机生成的随机数作为其初始值&#xff0c;通过 SYN 包传给接收端主机&#xff0c;每发送一次数据&#xff0c;就「累…

后端技术选型 sa-token校验学习 下 结合项目学习 后端鉴权

目录 后端注册拦截器 实现对 WebMvcConfigurer 接口的类实现 静态变量 方法重写 注册 Spring Framework拦截器 Sa-Token中SaServletFilter拦截器 思考 为什么使用两个拦截器 1. Spring Framework 拦截器 2. SaServletFilter 为什么要注册两个拦截器&#xff1f; 总结 …

使用 Charles 调试 Flutter 应用中的 Dio 网络请求

为了成功使用 Charles 抓取并调试 Flutter 应用程序通过 Dio 发起的网络请求&#xff0c;需遵循特定配置步骤来确保应用程序能够识别 Charles 的 SSL 证书&#xff0c;并正确设置代理服务器。 配置 Charles 以支持 HTTPS 请求捕获 Charles 默认会拦截 HTTP 流量&#xff1b;…

亿道三防丨三防笔记本是什么意思?和普通笔记本的优势在哪里?

三防笔记本是什么意思&#xff1f;和普通笔记本的优势在哪里&#xff1f; 在现代社会中&#xff0c;笔记本电脑已经成为人们工作和生活中不可或缺的一部分。然而&#xff0c;在一些特殊行业或环境中&#xff0c;普通笔记本电脑由于其脆弱性和对环境条件的敏感性&#xff0c;往…

Spring Boot教程之五十七:在 Apache Kafka 上发布 JSON 消息

Spring Boot | 如何在 Apache Kafka 上发布 JSON 消息 Apache Kafka是一个发布-订阅消息系统。消息队列允许您在进程、应用程序和服务器之间发送消息。在本文中&#xff0c;我们将了解如何在 Spring Boot 应用程序中向 Apache Kafka 发送 JSON 消息。 为了了解如何创建 Spring…

解决“无法定位程序输入点 av_buffer_create 于动态链接库 XXX\Obsidian.exe 上”问题

解决“无法定位程序输入点 av_buffer_create 于动态链接库 XXX\Obsidian.exe 上”问题 问题描述 本人在使用zotero中的zotero one&#xff08;青柠学术插件&#xff09;的时候&#xff0c;使用插件跳转obsidian中的对应笔记&#xff0c;出现上图情况。&#xff08;错误中提到的…

切削刀具热处理的作用学习笔记分享

对于一个搞冷加工的打工仔来说&#xff0c;热工的知识总是感觉那么新鲜。本期一起来学习一下切削刀具的热处理的一点点内容&#xff0c;虽然不是那么专业&#xff0c;但是了解一些还是很有好处的&#xff0c;废话不多说了&#xff0c;直接开始&#xff1a; 切削刀具热处理的作…

基于Web的宠物医院看诊系统设计与实现(源码+定制+开发)在线预约平台、宠物病历管理、医生诊疗记录、宠物健康数据分析 宠物就诊预约、病历管理与健康分析

博主介绍&#xff1a; ✌我是阿龙&#xff0c;一名专注于Java技术领域的程序员&#xff0c;全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师&#xff0c;我在计算机毕业设计开发方面积累了丰富的经验。同时&#xff0c;我也是掘金、华为云、阿里云、InfoQ等平台…

WPS excel使用宏编辑器合并 Sheet工作表

使用excel自带的工具合并Sheet表&#xff0c;我们会发现需要开通WPS会员才能使用合并功能&#xff1b; 那么WPS excel如何使用宏编辑器进行合并 Sheet表呢&#xff1f; 1、首先我们要看excel后缀是 .xlsx 还是 .xls &#xff1b;如果是.xlsx 那么 我们需要修改为 .xls 注…

ubuntu22.4 ROS2 安装gazebo(环境变量配置)

ubuntu版本&#xff1a;ubuntu22.4 最近在学习ROS2 视频教程古月居的入门课&#xff1a; 视频教程 文字笔记 问题 在学到关于Gazebo的时候&#xff0c;遇到下面问题&#xff1a; 运行 $ ros2 launch gazebo_ros gazebo.launch.py在这里卡住&#xff0c;不弹出gazebo 解决…

【Linux】7.Linux基础开发工具使用(1)

文章目录 1. Linux 软件包管理器 yum1.1 什么是软件包1.2 关于 rzsz1.3 查看软件包1.4 如何安装软件1.5 如何卸载软件我怎么知道要安装什么软件呢&#xff1f;源常用命令对照表&#xff1a; 2. Linux开发工具Linux编辑器-vim使用2.1 vim的基本概念命令模式&#xff1a;2.2 简单…

漫话架构师|什么是系统架构设计师(开篇)

~犬&#x1f4f0;余~ “我欲贱而贵&#xff0c;愚而智&#xff0c;贫而富&#xff0c;可乎&#xff1f; 曰&#xff1a;其唯学乎” 关注犬余&#xff0c;共同进步 技术从此不孤单