百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

Java操作Office:POI之word生成

liuian 2025-02-15 00:06 11 浏览

程序员架构进阶

一 背景

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

二 Java操作Office方案

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

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

三 ApachePOI

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 引用依赖


    8
    8
    4.1.0






    
        org.apache.poi
        poi
        ${poi.version}
    


    
    
        org.apache.poi
        poi-ooxml
        ${poi.version}
    
    
        org.apache.poi
        poi-excelant
        ${poi.version}
    


    
    
        com.deepoove
        poi-tl
        1.5.0
    


    
        org.projectlombok
        lombok
        1.18.20
    

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 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 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模板,然后用模板内容替换来生成复杂样式的表格。这个在后续文章中再做介绍,大家也可以先搜索相关的实现来学习了解。

相关推荐

Docker 47 个常见故障的原因和解决方法

【作者】曹如熙,具有超过十年的互联网运维及五年以上团队管理经验,多年容器云的运维,尤其在Docker和kubernetes领域非常精通。Docker是一种相对使用较简单的容器,我们可以通过以下几种方式...

电脑30个快问快答,解决常见电脑问题

1.强行关机/停电对电脑有影响吗?答:可能损坏硬盘(机械硬盘风险高)、未保存数据丢失,偶尔一次影响小,但频繁操作会缩短硬件寿命。2.C盘满影响速度吗?答:会!系统运行需C盘空间缓存临时数据,空间不...

使用Tcpdump包抓取分析数据包的详细用法

TcpDump可以将网络中传送的数据包的“头”完全截获下来提供分析。它支持针对网络层、协议、主机、网络或端口的过滤,并提供and、or、not等逻辑语句来帮助你去掉无用的信息。tcpdump就是一种...

电脑启动不了(BootDevice Not Found Hard Disk-3F0)解决方案

HP品牌机,开机启动不了,黑屏,开机取下主板电池恢复BIOS后,开机显示找不到启动盘。一、按F2键进入BIOS,出现硬盘内存检测界面的话,直接退出。就会出现这个界面,光标键向下,选择BIOSSetu...

电脑开机黑屏别慌!快码住!起底维修老师傅不能说的秘密

按下开机键却只收获黑屏大礼包?那些神秘的英文提示、刺耳的蜂鸣声,其实是电脑在给你发送求救信号!从按下电源到进入桌面的12秒里,你的电脑经历了史诗级的硬件自检与系统加载,今天我们就破译这段“摩斯电码”。...

电脑启动故障为何总要先看BIOS?新手必读的关键知识解析

最近在帮朋友们解答电脑无法正常开机的问题时,发现大家经常收到一句高频建议:“先检查BIOS”。对不少普通用户而言,BIOS依然是个神秘的存在。那么,BIOS到底是什么?电脑出现哪些故障会与它相关呢?本...

Windows 11 KB5053598更新:安全补丁还是系统噩梦?

2025年3月11日,微软发布了Windows1124H2的强制性更新KB5053598,作为“周二补丁日”(PatchTuesday)的一部分。然而,这款本应提升系统安全性的更新却引发了广泛的...

飞牛OS入门安装遇到问题,如何解决?

之前小编尝试了用旧电脑装飞牛OS安装之前特意查了一些硬件要求飞牛OS目前支持主流的x86架构硬件主机需能连网线飞牛OS暂时不支持只有无线网卡的安装貌似很多小伙伴在一开始安装就卡住了那今天咱们汇总分...

几种常见的电脑开机黑屏显示白色英文字母解决方法

当电脑开机出现黑屏并显示白色英文字母时,通常表示系统启动过程中遇到了错误。以下是几种常见原因及对应的解决方法,按照排查顺序整理:一、检查外接设备与硬件连接可能原因:外接U盘、移动硬盘等未拔出,或内部硬...

电脑启动出现问题,为什么都要先检查BIOS?

【ZOL中关村在线原创技巧应用】最近在回答问题的时候,总会发现很多朋友都在问“电脑无法正常开机怎么办?”这样类似的问题,而许多DIY大佬的回复总会出现一条高频建议“先检查BIOS”。但对于许多普通用户...

教你怎么用JavaScript检测当前浏览器是无头浏览器

什么是无头浏览器(headlessbrowser)?无头浏览器是指可以在图形界面情况下运行的浏览器。我可以通过编程来控制无头浏览器自动执行各种任务,比如做测试,给网页截屏等。为什么叫“无头”浏览器?...

12个高效的Python爬虫框架,你用过几个?

实现爬虫技术的编程环境有很多种,Java、Python、C++等都可以用来爬虫。但很多人选择Python来写爬虫,为什么呢?因为Python确实很适合做爬虫,丰富的第三方库十分强大,简单几行代码便可实...

运维的报表之路,用 node.js 轻松发送 grafana 报表

在运维过程中,无论是监控还是报表,都会有一些通过邮件发送图表的需求,由于开源的zabbix,grafana和kibana等并不完全具有“想发送哪儿就发送哪儿”的图片生成功能,在grafana...

C#基于浏览器内核的高级爬虫(c#爬取网页内容)

基于C#.NET+PhantomJS+Sellenium的高级网络爬虫程序。可执行Javascript代码、触发各类事件、操纵页面Dom结构、甚至可以移除不喜欢的CSS样式。很多网站都用Ajax动态加...

如何优化一个秒杀项目?(秒杀实现思路)

问题1:使用jmeter性能压测,定位瓶颈代码步骤流程:线程组--->Http请求--->查看结果树--->聚合报告tips:host的文件--->优先调用映射,减少DNS的时...