使用 poi-tl 生成Word文档的实例 - 多文档嵌套

使用poi-tl生成Word文档的实例

在本文中,我们将展示如何使用 poi-tl库 来生成包含数据的Word文档。

1. 添加依赖

<dependency>
    <groupId>com.deepoove</groupId>
    <artifactId>poi-tl</artifactId>
    <version>1.12.1</version>
</dependency>

2. 添加模版

在此之前先解释下:
• 在 poi-tl 中,文本内容使用 {{var}} 这样的双括号标识
• 文档嵌套,使用 {{+var}} 标识

tmp1.docx 模板文件内容如下:

{{title}}
来自:{{author}} 发布于:{{createTime}}

详细信息:
{{+subTmp}}

tmp2.docx 模板文件内容如下:

名称:{{name}}
年龄:{{age}}

3. 编写Java代码

package cn.yujky.io;

import com.deepoove.poi.XWPFTemplate;
import com.deepoove.poi.data.ChartSingleSeriesRenderData;
import com.deepoove.poi.data.Charts;
import com.deepoove.poi.data.Includes;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * @ClassName: DocTests
 * @Description:
 * @Author: yujky
 * @Date: 2023/7/18 15:11
 */
@SpringBootTest
public class DocTests {
    @Test
    public void testCreate() {
        try (
                XWPFTemplate tmp1 = XWPFTemplate.compile(new File("/Users/yujky/IdeaProjects/yujky-study/yujky-io/src/test/java/cn/yujky/io/tmp1.docx"));
        ) {
            tmp1.render(
                    new HashMap<String, Object>() {{
                        // 1. 添加tmp1文档的数据
                        put("title", "标题");
                        put("author", "yujky");
                        put("createTime", "2024-10-21 10:23:20");
                        // 2. 添加tmp2文档的数据
                        List<Map<String, Object>> subData = new ArrayList<>();
                        for (int i = 0; i < 10; i++) {
                            int finalI = i;
                            subData.add(new HashMap<String, Object>() {{
                                put("name", "yujky" + finalI);
                                put("age", finalI + 10);
                            }});
                        }
                        put("subTmp", Includes.ofLocal("/Users/yujky/IdeaProjects/yujky-study/yujky-io/src/test/java/cn/yujky/io/tmp2.docx").setRenderModel(subData).create());
                    }}
            );
            // 生成最终文档
            tmp1.writeAndClose(Files.newOutputStream(Paths.get("/Users/yujky/IdeaProjects/yujky-study/yujky-io/src/test/java/cn/yujky/io/tmp1Res.pdf")));
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

4. 运行代码

运行上述Java代码,将生成最终的文档,并保存为 tmp1Res.docx

5. 结果展示

生成的 Word 文档内容如下:

标题
来自:yujky 发布于:2024-10-21 10:23:20

详细信息:
名称:yujky0
年龄:10
名称:yujky1
年龄:11
......(省略部分内容)
名称:yujky9
年龄:19

通过上述步骤,即可成功地使用poi-tl库来生成包含数据的Word文档,并实现了模板之间的嵌套和渲染

官方参考文档:嵌套 https://deepoove.com/poi-tl/#_%E5%B5%8C%E5%A5%97

### poi-tl 区块对树结构的处理 在 `poi-tl` 中,区块对用于灵活地控制文档元素的渲染次数。对于树结构的数据模型,可以通过嵌套区块来实现层次化的数据展示。 #### 嵌套区块表示树节点 假设有一个树形结构的数据集,其中每个节点可能有子节点。通过定义多个层级的区块,可以在模板中逐层遍历并渲染这些节点: ```html {{#treeNode}} <p>${name}</p> {{#children}} <!-- 子节点重复此模式 --> <ul> <li>{{> treeNode}}</li> </ul> {{/children}} {{/treeNode}} ``` 上述代码片段展示了如何利用区块对 `${}` 和结束标记 `/` 来迭代显示树状列表[^1]。这里的关键在于递归调用相同的区块名称 (`{{> treeNode}}`) 实现自相似性的表达。 #### 动态调整区块逻辑 当面对更复杂的场景时,比如某些特定条件下才展开某个分支,则可在 Spring EL 表达式的支持下加入条件判断语句: ```html {{#if showChildren && children.size() > 0 }} <details open="open"> <summary>${title}</summary> <ol> {{#each children as child}} <li>{{child.name}}</li> {{/each}} </ol> </details> {{/if}} ``` 这段示例说明了如何结合布尔属性 `showChildren` 控制是否呈现其下的子项集合[^2]。注意这里的语法细节:使用了 HTML5 的 `<details>` 标签配合 JavaScript 风格的伪类选择器 `.size()` 方法获取数组长度。 为了确保资源管理的安全性和效率,在完成 Word 文档创建之后应当及时释放所占用的各种输入输出流对象: ```java POIUtils.closeQuietlyMulti(template, byteArrayOutputStream, fileOutputStream); ``` 该方法能够优雅地关闭所有给定参数里的可关闭实例而不会抛出异常中断程序执行流程[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值