写点什么

Java 操作 Office:POI 之 word 生成

发布于: 2 小时前
Java操作Office:POI之word生成

一 背景

最近在项目开发中,有数据导出到 word 的需求。这就涉及代码生成 word 文档的操作,且有格式要求。 大家用 word 做过简历的都有了解,做简历时,会使用表格、图片、文字等元素。而且表格也可能有嵌套、合并单元格,以及插入图片到单元格的操作。该怎么做?

二 Java 操作 Office 方案

百度一下 Java Office 操作,或者再直接一点搜索 Java word,就比较容易搜到 iText、POI 等组件。在文章 Java导出word的几种方式 这篇文章中,提到了包括 Jacob、Apache POI、Java2word、iText、FreeMarker 五种方式。

通过对比,结合需求要求,最终选择了 Apache POI 来实现,所以这里先详细介绍 POI,以及一个可用的 demo,供参考。

三 Apache POI

Apache POI(官网)是基于 Office Open XML 标准(OOXML)和 Microsoft 的 OLE 2 复合文档格式(OLE2)处理各种文件格式的开源项目。 简而言之,您可以使用 Java 读写 MS Excel 文件,可以使用 Java 读写 MS Word 和 MS PowerPoint 文件。

poi 的 gitee 地址:gitee。入门教程可以参考 Apache POI Word(docx) 入门示例教程

四 版本信息

poi 的最新版本已经到了 5.0.0,不过可以找到的大部分 demo 都是基于 3.x 版本或 4.1 版本。为了尽快搭建 demo 并运行起来,我们也没有使用最新版本,而是选择了 4.1.0 进行开发。

4.1 引用依赖

<properties>    <maven.compiler.source>8</maven.compiler.source>    <maven.compiler.target>8</maven.compiler.target>    <poi.version>4.1.0</poi.version></properties>

<dependencies> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>${poi.version}</version> </dependency>
<!-- poi处理xlsx格式,用于处理word中的表格 --> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>${poi.version}</version> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-excelant</artifactId> <version>${poi.version}</version> </dependency>
<!-- poi-tl基于poi的word模板引擎 --> <dependency> <groupId>com.deepoove</groupId> <artifactId>poi-tl</artifactId> <version>1.5.0</version> </dependency>
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.18.20</version> </dependency></dependencies>
复制代码

4.2 创建 word 示例代码

4.2.1 创建新的文档

创建 word 文档比较简单,直接使用 new XWPFDocument 即可,XWPFDocument 是对 .docx 文档操作的高级封装 API:

XWPFDocument doc = new XWPFDocument();
复制代码

4.2.2 表格

即 Word 文档中的表格。API 创建时需要指定行数和列数,示例如下:

//创建一个表格,并指定宽度XWPFTable table = doc.createTable(4, 4);TableTools.widthTable(table, MiniTableRenderData.WIDTH_A4_FULL, 4);
//设置第0行数据List<XWPFTableCell> row0 = table.getRow(0).getTableCells();row0.get(0).setText("xxxx"); //为第0行第0列设置内容row0.get(0).setWidth("200");row0.get(1).setText("aaaa");row0.get(2).setText("bbbb");row0.get(3).setText("cccc");
复制代码

常规的简单表格,我们只要按照上述代码逐行操作即可;但现实中不会这么容易。通常会涉及在单元格插入图片、合并行、合并列,甚至表格嵌套。目前表格嵌套暂未实现,先介绍其他三种情况。

4.2.3 列合并

有两种方法,一种是使用 addNewHMerge 方法,通过设置合并的起始列和结束列,逐个列进行合并:

List<XWPFTableCell> row2_1 = table.getRow(2).getTableCells();row2_1.get(0).setText("合并表格"); //为第0行第0列设置内容//将第一列到第四列合并for (int i = 1; i <= 3; i++) {    //对单元格进行合并的时候,要标志单元格是否为起点,或者是否为继续合并    if (i == 1)        row2_1.get(i).getCTTc().addNewTcPr().addNewHMerge().setVal(STMerge.RESTART);//这是起点    else        row2_1.get(i).getCTTc().addNewTcPr().addNewHMerge().setVal(STMerge.CONTINUE);//继续合并}
复制代码

在业务代码中这样的写法稍显繁琐,我们也可以直接使用 TableTools.mergeCellsHorizonal()函数来执行合并:

// 合并第一行的第0列到第8列单元格TableTools.mergeCellsHorizonal(table, 1, 0, 8);
复制代码


4.2.4 行合并

如果是要合并某几行,也可以使用 TableTools 提供的方法:

// 合并第0列的第一行到第九行的单元格TableTools.mergeCellsVertically(table, 0, 1, 9);
复制代码

我们看一下 TableTools 的源码:

public static void mergeCellsVertically(XWPFTable table, int col, int fromRow, int toRow) {        if (toRow > fromRow) {            for(int rowIndex = fromRow; rowIndex <= toRow; ++rowIndex) {                XWPFTableCell cell = table.getRow(rowIndex).getCell(col);                CTTcPr tcPr = getTcPr(cell);                CTVMerge vMerge = tcPr.addNewVMerge();                if (rowIndex == fromRow) {                    vMerge.setVal(STMerge.RESTART);                } else {                    vMerge.setVal(STMerge.CONTINUE);                }            }
} }
复制代码

可以发现,底层还是使用 addNewVMerge 等方法,也设置了起始和结束位置,只是做了一层封装。

4.2.5 图片插入表格

图片插入表格要麻烦一些,如果大家在百度上搜过插入图片到表格方法,大概率会找到这样的操作:

大部分对应的都是 3.9 以前的版本,写起来比较复杂,而且在 4.x 之后,图中 super.getRelationId()方法也发生了变化,代码报错。

通过调研,发现 XWPFRun 中提供了 addPicture 方法,写起来也简单了很多。一个示例如下:

String imageFile = "/Users/xxx/Downloads/图片 1.png";InputStream stream = new FileInputStream(imageFile);
//表格中创建段落XWPFParagraph paragraph = row2_1.get(1).getParagraphs().get(0);XWPFRun run = paragraph.createRun();run.addPicture(stream, XWPFDocument.PICTURE_TYPE_PNG, "Generated", Units.toEMU(364), Units.toEMU(256));
复制代码

run.addPicture 接收的参数依次为:图片的 InputStream 流,图片类型,图片名称(非文件名),图片宽度、图片高度。通过这个方法,我们就可以把图片插入到指定的表格中,并设置图片的宽高属性。

五 总结

通过上述介绍,大家应该可以简单实现一个表格了。本文的方式还是偏向于硬编码的方式,在很多场景(例如简历、报表等典型场景)可以采用模板的方式,创建 word 模板,然后用模板内容替换来生成复杂样式的表格。这个在后续文章中再做介绍,大家也可以先搜索相关的实现来学习了解。

发布于: 2 小时前阅读数: 4
用户头像

磨炼中成长,痛苦中前行 2017.10.22 加入

微信公众号【程序员架构进阶】。多年项目实践,架构设计经验。曲折中向前,分享经验和教训

评论

发布
暂无评论
Java操作Office:POI之word生成