SpringBoot+EasyPOI轻松实现Excel和Word导出PDF

2025-07-18 19:50

本文主要是介绍SpringBoot+EasyPOI轻松实现Excel和Word导出PDF,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《SpringBoot+EasyPOI轻松实现Excel和Word导出PDF》在企业级开发中,将Excel和Word文档导出为PDF是常见需求,本文将结合​​EasyPOI和​​Aspose系列工具实...

在企业级开发中,将 Excel 和 Word 文档导出为 PDF 是常见需求(如报表归档、合同存档)。本文将结合 ​​EasyPOI(Excel 快速导出)​​ 和 ​​Aspose 系列工具(格式完美转换)​​,详细讲解如何实现 Excel 和 Word 到 PDF 的转换,并解决格式保留、性能优化等核心问题。

一、环境准备与依赖配置

1.1 方案选型

步骤工具/库特点适用场景
​​Excel 生成​​EasyPOI(阿里)基于 Apache POI 优化,支持注解快速生成 Excel,内存占用低数据量中等(十万行以内)
​​Excel 转 PDphpF​​Aspose.Cells(商业)完美保留 Excel 格式(字体、颜色、合并单元格、图表等),性能强企业级高精度转换需求
​​Word 生成​​Apache POI(XwpF)开源,支持 .docx 格式,需手动处理复杂样式简单 Word 文档(文本+表格)
​​Word 转 PDF​​Aspose.Words(商业)完美保留 Word 格式(样式、图片、页眉页脚等),兼容性强企业级高精度转换需求

1.2 依赖配置(商业库方案)

​注意​​:Aspose 系列工具需购买商业许可证(试用版有水印),测试阶段可使用 免费试用版。

pom.XML 中添加依赖:

<dependencies>
    <!-- Spring Boot Web -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    
    <!-- EasyPOI(Excel 导出) -->
    <dependency>
        <groupId>cn.afterturn</groupId>
        <artifactId>easypoi-base</artifactId>
        <version>4.4.0</version> <!-- 最新稳定版 -->
    </dependency>
    <dependency>
        <groupId>cn.afterturn</groupId>
        <artifactId>easypoi-web</artifactId>
        <version>4.4.0</version>
    </dependency>
    
    <!-- Aspose.Cells(Excel 转 PDF) -->
    <dependency>
        <groupId>com.aspose</groupId>
        <artifactId>aspose-cells</artifactId>
        <version>23.10</version>
   http://www.chinasem.cn </dependency>
    
    <!-- Aspose.Words(Word 转 PDF) -->
    <dependency>
        <groupId>com.aspose</groupId>
        <artifactId>aspose-words</artifactId>
        <version>23.10</version>
    </dependency>
</dependencies>

二、Excel 导出 PDF 实现(EasyPOI + Aspose.Cells)

2.1 定义 Excel 实体类(EasyPOI 注解)

使用 EasyPOI 的注解简化 Excel 生成逻辑,支持标题、表头、数据校验等:

@Data
@TableName("sys_employee")
// Excel 导出配置(文件名、工作表名)
@ExcelEntity(name = "员工信息表", sheetName = "员工信息")
public class Employee {
    @Excel(name = "ID", orderNum = "0", width = 10)  // 列名、顺序、列宽
    private Long id;

    @Excel(name = "姓名", orderNum = "1", width = 15)
    private String name;

    @Excel(name = "年龄", orderNum = "2", width = 10, replace = {"0_未知", "1_男", "2_女"})  // 数据替换
    private Integer gender;
China编程
    @Excel(name = "部门", orderNum = "3", width = 20)
    private String department;

    @Excel(name = "入职时间", orderNum = "4", width = 20, format = "yyyy-MM-dd")  // 日期格式
    private Date hireDate;
}

2.2 Excel 生成工具类(EasyPOI)

编写工具类,封装 EasyPOI 的 Excel 导出逻辑,生成 Workbook 对象:

@Component
public class ExcelExportUtils {
    private static final Logger log = LoggerFactory.getLogger(ExcelExportUtils.class);

    /**
     * 生成 Excel Workbook(基于 EasyPOI)
     * @param dataList 数据列表
     * @param entityClass 实体类(含 @ExcelEntity 注解)
     * @return Workbook
     */
    public static <T> Workbook generateExcelWorkbook(List<T> dataList, Class<T> entityClass) {
        ExportParams exportParams = new ExportParams();
        exportParams.setSheetName("员工信息");  // 工作表名(与 @ExcelEntity 一致)
        exportParams.settitle("员工信息表");    // Excel 标题(顶部大标题)

        // 使用 EasyPOI 生成 Workbook
        return ExcelExportUtil.exportExcel(exportParams, entityClass, dataList);
    }
}

2.3 Excel 转 PDF 工具类(Aspose.Cells)

使用 Aspose.Cells 加载 Excel 的 Workbook,并转换为 PDF 字节流,保留所有格式:

@Component
public class ExcelToPdfConverter {
    private static final Logger log = LoggerFactory.getLogger(ExcelToPdfConverter.class);

    /**
     * 将 Workbook 转换为 PDF 字节流(Aspose.Cells)
     * @param workbook Excel Workbook
     * @return PDF 字节流
     */
    public static ByteArrayOutputStream convertToPdf(Workbook workbook) throws Exception {
        // 1. 创建 PDF 保存选项(可选配置)
        PdfSaveOptions saveOptions = new PdfSaveOptions();
        saveOptions.setOnePagePerSheet(true);       // 每个工作表一页
        saveOptions.setFormat(PdfFormat.PdfA2B);    // 符合 PDF/A-2B 标准(可选)
        saveOptions.setCompressImages(true);        // 压缩图片(减小文件大小)
        saveOptions.setImageCompression(PdfImageCompressionType.MEDIUM);  // 图片压缩级别

        // 2. 转换为 PDF 字节流
        ByteArrayOutputStream pdfOutputStream = new ByteArrayOutputStream();
        workbook.save(pdfOutputStream, saveOptions);
        return pdfOutputStream;
    }
}

2.4 Controller 层整合(Excel 导出 PDF)

编写接口,接收请求后生成 Excel 并转换为 PDF 下载:

@RestController
@RequestMapping("/export")
public class ExportController {
    @Autowired
    private ExcelExportUtils excelExportUtils;
    @Autowired
    private ExcelToPdfConverter pdfConverter;

    @GetMapping("/excel-to-pdf")
    public void exportEmployeeToPdf(HttpServletResponse response) throws Exception {
        // 1. 模拟数据(实际从数据库查询)
        List<Employee> dataList = new ArrayList<>();
        dataList.add(new Employee(1L, "张三", 25, "技术部", new Date()));
        dataList.add(new Employee(2L, "李四", 30, "市场部", new Date()));

        // 2. 生成 Excel Workbook(EasyPOI)
        Workbook workbook = excelExportUtils.generateExcelWorkbook(dataList, Employee.class);

        // 3. 转换为 PDF 字节流(Aspose.Cells)
        ByteArrayOutputStream pdfOutputStream = pdfConverter.convertToPdf(workbook);

        // 4. 设置响应头(下载 PDF)
        response.setContentType("application/pdf");
        response.setHeader("Content-Disposition", "attachment; filename=employee_report.pdf");
        response.getOutputStream().write(pdfOutputStream.toByteArray());
        response.getOutputStream().flush();
        response.getOutputStream().close();
    }
}

三、Word 导出 PDF 实现(EasyPOI + Aspose.Words)

3.1 定义 Word 模板(Freemarker 或直接生成)

Word 文档生成通常使用 ​​模板引擎​​(如 Freemarker),通过替换模板中的占位符生成内容。这里以 Aspose.Words 直接生成简单 Word 为例:

@Component
public class WordDocumentBuilder {
    private static final Logger log = LoggerFactory.getLogger(WordDocumentBuilder.class);

    /**
     * 构建 Word 文档(Aspose.Words)
     * @param content 文档内容(文本+表格)
     * @return Document(Aspose.Words 文档对象)
     */
    public static Document buildwordDocument(String content) throws Exception {
        // 创建空白文档
        Document doc = new Document();
        DocumentBuilder builder = new DocumentBuilder(doc);

        // 写入标题
        builder.getFont().setName("微软雅黑");
        builder.getFont().setSize(16);
        builder.getFont().setBold(true);
        builder.writeln("员工信息报告");

        // 写入正文
        builder.getFont().setSize(12);
        builder.getFont().setBold(false);
        builder.writeln("以下是近期员工信息:");
        builder.writeln();

        // 写入表格(示例数据)
        builder.startTable();
        builder.insertCell();
        builder.write("ID");
        builder.insertCell();
        builder.write("姓名");
        builder.insertCell();
        builder.write("部门");
        builder.endRow();

        builder.insertCell();
        builder.write("1");
        builder.insertCell();
        builder.write("张三");
        builder.insertCell();
        builder.write("技术部");
        builder.endRow();

        builder.endTable();

        return doc;
    }
}

3.2 Word 转 PDF 工具类(Aspose.Words)

使用 Aspose.Words 将生成的 Word 文档转换为 PDF,保留样式、图片等:

@Component
public class WordToPdfConverter {
    private static final Logger log = LoggerFactory.getLogger(WordToPdfConverter.class);

    /**
     * 将 Aspose.Words Document 转换为 PDF 字节流
     * @param document Word 文档对象
     * @return PDF 字节流
     */
    public static ByteArrayOutputStream convertToPdf(Document document) throws Exception {
        // 1. 配置 PDF 保存选项(可选)
        PdfSaveOptions saveOptions = new PdfSaveOptions();
        saveOptions.setCompressFonts(true);       // 压缩字体
        saveOptions.setUseHighQualityRendering(true);  // 高质量渲染

        // 2. 转换为 PDF 字节流
        ByteArrayOutputStream pdfOutputStream = new ByteArrayOutputStream();
        document.save(pdfOutputStream, saveOptions);
        return pdfOutputStream;
    }
}

3.3 Controller 层整合(Word 导出 PDF)

编写接口,生成 Word 文档并转换为 PDF 下载:

@RestController
@RequestMapping("/export")
public class ExportController {
    @Autowired
    private WordDocumentBuilder wordBuilder;
    @Autowired
    private WordToPdfConverter wordPdfConverter;

    @GetMapping("/word-to-pdf")
    public void exportEmployeeToPdf(HttpServletResponse response) throws Exception {
        // 1. 构建 Word 文档(Aspose.Words)
        String content = "员工信息报告内容...";  // 实际从数据库或业务逻辑获取
        Document wordDoc = wordBuilder.buildWordDocument(content);

        // 2. 转换为 PDF 字节流(Aspose.Words)
        ByteArrayOutputStream pdfOutputStream = wordPdfConverter.convertToPdf(wordDoc);

        // 3. 设置响应头(下载 PDF)
        response.setContentType("application/pdf");
        response.setHeader("Content-Disposition", "attachment; filename=employee_report.pdf");
        response.getOutputStream().write(pdfOutputStream.toByteArray());
        response.getOutputStream().flush();
        response.getOutputStream().close();
    }
}

四、关键问题与优化

4.1 格式保留问题

  • ​Excel​​:Aspose.Cells 支持所有 Excel 特性(合并单元格、条件格式、图表、公式等),转换后与原文档一致。
  • ​Word​​:Ashttp://www.chinasem.cnpose.Words 支持样式(字体、颜色、段落)、图片、页眉页脚、目录等,复杂排版(如多栏、文本框)也能完美保留。

4.2 性能优化

  • ​大数据量 Excel​​:若数据量超过 10 万行,建议使用 EasyPOI 的 @ExcelEntity 配合分页查询,避免内存溢出。
  • ​PDF 压缩​​:通过 PdfSaveOptions 配置图片压缩(PdfImageCompressionType)和字体嵌入策略,减小文件体积。

4.3 免费方案替代(LibreOffice)

若无法使用商业库,可通过调用 LibreOffice 命令行实现转换(需服务器安装 LibreOffice):

// 调用 LibreOffice 转换 Excel 到 PDF
public static void convertExcelToPdfWithLibreOffice(String excelPath, String pdfPath) throws IOException {
    String command = String.format(
        "libreoffice --headless --convert-to pdf --outdir %s %s",
        pdfPath, excelPath
    );
    Process process = Runtime.getRuntime().exec(command);
    int exitCode = process.waitFor();
    if (exitCode != 0) {
        throw new RuntimeException("LibreOffice 转换失败");
    }
}

​注意​​:此方法格式保留可能不完整(如复杂样式丢失),仅适用于小型项目。

五、总结

本文通过 ​​EasyPOI 生成 Excel/WordBiljwZT​ + ​​Aspose 系列工具转换 PDF​​ 的方案,实现了 Spring Boot 中 Excel 和 Word 到 PDF 的完整流程。Aspose 工具在格式保留和性能上表现优异,适合企业级高精度需求;若预算有限,可尝试 LibreOffice 命令行方案(需接受格式限制)。实际项目中需根据数据量和格式要求选择合适方案。

以上就是SpringBoot+EasyPOI轻松实现Excel和Word导出PDF的详细内容,更多关于SpringBoot Excel和Word导出PDF的资料请关注编程China编程(www.chinasem.cn)其它相关文章!

这篇关于SpringBoot+EasyPOI轻松实现Excel和Word导出PDF的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三