根據模板動態生成word(一)使用freemarker生成word

来源:https://www.cnblogs.com/fhey/archive/2023/07/09/17539412.html
-Advertisement-
Play Games

如果模板里需要用變數填充表格,建議模板里的表格像word文件一樣建一個兩行的表格。但是這樣是freemaker是無法成功替換變數的,所以需要手動處理成到一個段里(如圖2),關於這點實在太無語了,因為沒有找到比較好的處理辦法,只能手工處理,在實際的開發工作中曾經花了幾個小時來做這件事情。根據模板文件生... ...


@

目錄

一、準備模板

1、創建模板文件

首先先建立一個word文件,輸入模板內容freemaker的內容,下麵是本次演示的word文件。
docx格式演示word模板

然後將word文件另存為 .xml 文件,然後再把文件尾碼改成.ftl 。將項目的resource目錄下建立一個templates目錄(非必須步驟)將 模板文件放到templates目錄下
請添加圖片描述

請添加圖片描述
打開模板文件按 Ctrl + Shift + L 將模板內容格式化。

2、處理模板

2.1 處理普通文本

處理文本比較簡單,將需要替換文本中直接用占位符 ${} 替換即可。

這裡發現一個問題因為之前在word格式時我就已經替換了變數,但是在ftl變數卻被 拆分成多段了(見圖1)。但是這樣是freemaker是無法成功替換變數的,所以需要手動處理成到一個段里(如圖2),關於這點實在太無語了,因為沒有找到比較好的處理辦法,只能手工處理,在實際的開發工作中曾經花了幾個小時來做這件事情。
圖1:
請添加圖片描述

圖2
請添加圖片描述

2.2 處理表格

如果模板里需要用變數填充表格,建議模板里的表格像word文件一樣建一個兩行的表格。在模板中<<w:tbl>> 表示一個表格 、<w: tr> 表示一行、<w: tc> 表示一列。因為FreeMarker 是利用列表一行一行迴圈填充的,所以我們可以根據關鍵字找到<<w:tbl>>標簽,因為第一個 <w: tr>是表頭註意不要改到了,找到第二個<w: tr>在前後分別加上如下語句即可,後面的表格裡後面的行<w: tr>需要刪掉,建議模板里的表格像word文件一樣建一個兩行的表格即可這樣就不用刪了:

<#list itemList as item>
</#list>

替換後的模板如下:
請添加圖片描述

2.3 處理圖片

如果模板里需要用變數填充圖片,建議先在word文件里插入一張圖片,這樣在模板文件里找到<pkg:binaryData>標簽直接裡面把裡面的圖片base64字元替換成變數即可,word里可以通過植入base64字元來展示圖片:

替換前:
請添加圖片描述

替換後:

<pkg:binaryData>${image1}</pkg:binaryData>

到此模板已經調整完成,接下來就可以開始寫代碼了。

二、項目代碼

1、引入依賴

在項目的pom文件里引入如下依賴

	<dependency>
          <groupId>org.freemarker</groupId>
          <artifactId>freemarker</artifactId>
          <version>2.3.31</version>
      </dependency>

2、生成代碼

將圖片轉成Base64字元串的公共方法:

public static String getImageBase64Str(String imgFile) {
        try( InputStream in = new FileInputStream(imgFile)) {
            byte[] data = new byte[in.available()];
            in.read(data);
            BASE64Encoder encoder = new BASE64Encoder();
            return encoder.encode(data);
        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

根據模板文件生成word,主要生成的word的文件尾碼必須是doc不能是docx,不然生成的文件無法打開。

public static void crateWord(Map<String, Object> dataMap, String templatePath, String targetFile){
        String path = templatePath.substring(0,templatePath.lastIndexOf("/"));
        String templateName = templatePath.substring(templatePath.lastIndexOf("/") + 1);
        try (FileOutputStream out = new FileOutputStream(targetFile);
             Writer writer = new BufferedWriter(new OutputStreamWriter(out, "utf-8"))){
            Configuration configuration = new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS);
            configuration.setDefaultEncoding("utf-8");
            configuration.setClassForTemplateLoading(FreeMakerTest.class, path);
            //除了ClassForTemplateLoading外,另一種模板載入方式DirectoryForTemplateLoading,也可用
            //ClassPathResource resource = new ClassPathResource(path);
            //configuration.setDirectoryForTemplateLoading(resource.getFile());
            //載入模板
            Template template = configuration.getTemplate(templateName);
            //渲染模板
            template.process(dataMap, writer);
        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);
        } catch (UnsupportedEncodingException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            throw new RuntimeException(e);
        } catch (TemplateException e) {
            throw new RuntimeException(e);
        }
    }

三、驗證生成word

新建的列表數據實體類:

public class Arrears{
    private String name;
    private Integer num;

    private String endDay;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getNum() {
        return num;
    }

    public void setNum(Integer num) {
        this.num = num;
    }

    public String getEndDay() {
        return endDay;
    }

    public void setEndDay(String endDay) {
        this.endDay = endDay;
    }
}

準備模板填充數據

private static Map<String, Object> prepareParam(){
        LocalDate currentDate = LocalDate.now();
        LocalDate endDate = currentDate.plusYears(1L);
        List<Arrears> arrearsList = new ArrayList<>();
        arrearsList.add(new Arrears(){{setName("一頓老魏");setNum(1);setEndDay("三月內");}});
        arrearsList.add(new Arrears(){{setName("貴州大黃牛");setNum(1);setEndDay("一年內");}});
        arrearsList.add(new Arrears(){{setName("v我50");setNum(1);setEndDay("一月內");}});

        //填充所需要的數據
        Map<String, Object> dataMap = new HashMap<>();
        dataMap.put("debtor", "陳有楚");
        dataMap.put("nowYear", String.valueOf(currentDate.getYear()));
        dataMap.put("nowMonth", String.valueOf(currentDate.getMonthValue()));
        dataMap.put("nowDay", String.valueOf(currentDate.getDayOfMonth()));
        dataMap.put("arrears", "一頓老魏、貴州大黃牛、v我50");
        dataMap.put("endYear", String.valueOf(endDate.getYear()));
        dataMap.put("endMonth", String.valueOf(endDate.getMonthValue()));
        dataMap.put("endDay", String.valueOf(endDate.getDayOfMonth()));
        dataMap.put("arrearsList", arrearsList);
        dataMap.put("image1", getImageBase64Str("D:\\picture\\其他\\24-05-23-142418.png"));
        dataMap.put("creditor", "知北游");
        return dataMap;
    }

測試代碼:

public static void main(String[] args) throws IOException {
        //準備參數
        Map<String, Object> dataMap = prepareParam();
        crateWord(dataMap,"/templates/qiantiao.ftl","D:\\test\\qiantiao.doc");
    }

測試結果:
請添加圖片描述


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ![](https://img2023.cnblogs.com/blog/3076680/202307/3076680-20230704152811386-132747394.png) # 1. 記錄日誌 ## 1.1. 傳統的日誌文件仍然是最可靠和最靈活的信息載體 ## 1.2. 日誌文件反映應用 ...
  • ## 引言 **觀察者模式**是一種行為型設計模式,它允許對象之間建立一種一對多的關係,使得當一個對象狀態改變時,所有依賴它的對象都能夠自動得到通知並更新自己的狀態。該模式可以幫助我們實現松耦合的系統,以便更好地應對變化和擴展。 在觀察者模式中,有兩個角色:**觀察者**和**被觀察者**。被觀察者 ...
  • ## 引言 **工廠方法模式**是一種創建型設計模式,它定義了一個用於創建對象的介面,但是讓子類決定將哪一個類實例化。換句話說,工廠方法模式讓一個類的實例化延遲到其子類。 工廠方法模式有以下幾個主要角色: - 抽象工廠(AbstractFactory):聲明用於創建抽象產品的操作的介面。 - 工廠實 ...
  • ## 引言 **抽象工廠模式**一種創建型設計模式,它提供了一種方式來封裝一組具有相同主題的工廠,而不必指定它們具體的類。這樣,客戶端代碼就可以使用抽象工廠來創建一組相關的對象,而不必關心實際創建的具體類。 抽象工廠模式有以下幾個主要角色: - 抽象工廠(AbstractFactory):聲明用於創 ...
  • ### 歡迎訪問我的GitHub > 這裡分類和彙總了欣宸的全部原創(含配套源碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### 為什麼要編譯nginx-clojure源碼 - 作為《 ...
  • 大家好,我是 god23bin,今天繼續說 Spring 的內容,關於 Spring 中 Bean 的配置的,通過上一篇文章的學習,我們知道了 Spring 中的依賴註入,其中有兩種主要的方式,分別是基於構造方法的 DI 和 基於 Setter 的 DI。 ...
  • > 最近燒哥發現個寶藏項目,竟然用Java開發了暗黑2出來。 眾所周知,暗黑2是暴雪開發的一款經典游戲,距今雖有20多年,仍然有很多粉絲。 粉絲延續熱情的方式有很多,一種是做Mod,比如[魔電](https://www.median-xl.com/),對怪物、技能、物品、場景、甚至游戲機制都有大改, ...
  • # Qt deleteLater作用及源碼分析 > 個人經驗總結,如有錯誤或遺漏,歡迎各位大佬指正 🥳 在本篇文章中,我們將深入分析源碼,探討`deleteLater`的原理。 `deleteLater`是Qt框架提供的一個重要函數,用於在事件迴圈中延遲刪除對象。 在軟體開發中,延遲刪除對象的概念 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...