通过操作书签可以实现 word 模板替换变量的功能场景,本文一下代码内容,直接可以复制使用正常编译运行。

添加 maven 依赖

    <dependencies>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>5.2.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-examples</artifactId>
            <version>5.2.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-scratchpad</artifactId>
            <version>5.2.2</version>
        </dependency>
    </dependencies>

操作书签的代码

package org.example;

import org.apache.poi.hwpf.HWPFDocument;
import org.apache.poi.hwpf.usermodel.Bookmark;
import org.apache.poi.hwpf.usermodel.Bookmarks;
import org.apache.poi.hwpf.usermodel.Range;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.apache.poi.xwpf.usermodel.XWPFParagraph;
import org.apache.poi.xwpf.usermodel.XWPFRun;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.CTBookmark;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.CTP;
import org.w3c.dom.Node;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class BookmarkDemo {

    /**
     * 因为docx为xml格式的结构,一下为docx中定义的部分常量引用
     **/
    public static final String RUN_NODE_NAME = "w:r";
    public static final String TEXT_NODE_NAME = "w:t";
    public static final String BOOKMARK_START_TAG = "w:bookmarkStart";
    public static final String BOOKMARK_END_TAG = "w:bookmarkEnd";
    public static final String BOOKMARK_ID_ATTR_NAME = "w:id";
    public static final String STYLE_NODE_NAME = "w:rPr";


    public static void main(String[] args) throws IOException {
        // 读取 doc 文件中的所有书签
        InputStream inputStream = Files.newInputStream(Paths.get("D:\\Desktop\\Welcome2.doc"));
        getBookmarksByDoc(inputStream);

        // 读取 doc 文件中的所有书签
        InputStream inputStream2 = Files.newInputStream(Paths.get("D:\\Desktop\\Welcome.docx"));
        getBookmarksByDocx(inputStream2);

        // 替换文件中的 bookmark 内容
        InputStream inputStream3 = Files.newInputStream(Paths.get("D:\\Desktop\\Welcome.docx"));
        Map<String, String> dataMap = new HashMap<>();
        dataMap.put("strong", "单红宇");
        dataMap.put("footnotes", "李小雨");
        replaceBookmarksByDocx(inputStream3, Files.newOutputStream(Paths.get("D:\\Desktop\\Welcome3.docx")), dataMap);
    }


    /**
     * 读取 doc 文件中的所有书签
     *
     * @param inputStream
     * @throws IOException
     */
    public static void getBookmarksByDoc(InputStream inputStream) throws IOException {
        HWPFDocument wordDoc = new HWPFDocument(inputStream);

        Bookmarks bookmarks = wordDoc.getBookmarks();

        for (int b = 0; b < bookmarks.getBookmarksCount(); b++) {
            Bookmark bookmark = bookmarks.getBookmark(b);
            String bookMarkText = new Range(bookmark.getStart(), bookmark.getEnd(), wordDoc).text();

            System.out.println("[" + bookmark.getStart() + "; "
                    + bookmark.getEnd() + "]: " + bookmark.getName() + " = " + bookMarkText);
        }

    }

    /**
     * 读取 docx 文件中的所有书签
     *
     * @param inputStream
     * @throws IOException
     */
    public static void getBookmarksByDocx(InputStream inputStream) throws IOException {
        XWPFDocument docx = new XWPFDocument(inputStream);

        List<XWPFParagraph> paragraphList = docx.getParagraphs();
        for (XWPFParagraph xwpfParagraph : paragraphList) {
            CTP ctp = xwpfParagraph.getCTP();

            for (int dwI = 0; dwI < ctp.sizeOfBookmarkStartArray(); dwI++) {
                CTBookmark bookmark = ctp.getBookmarkStartArray(dwI);
                String bookmarkName = bookmark.getName();

                // 因为 docx 的结构是xml格式,它不像 doc 文档那样有具体的start和end值来定位一个bookmark的值范围,
                // 所有如果你想要读取 docx 文档中bookmark的内容,你需要从 bookmarkStart 节点开始逐级逐层依次解析xml文
                // 件的 nodeValue 值并进行拼接,直至读取到下一个名为 bookmarkEnd 的节点为止。
                // 下面这个方法 getBookmarkTextContent() 是一个没有实现的方法,如有需要请自行实现
                // String bookmarkTextContent = getBookmarkTextContent(bookmark);

                System.out.println(bookmarkName);
            }
        }
    }


    /**
     * docx 文件中书签的替换
     *
     * @param inputStream
     * @param outputStream
     * @param dataMap
     * @throws IOException
     */
    public static void replaceBookmarksByDocx(InputStream inputStream, OutputStream outputStream, Map<String, String> dataMap) throws IOException {
        XWPFDocument document = new XWPFDocument(inputStream).getXWPFDocument();
        List<XWPFParagraph> paragraphList = document.getParagraphs();
        for (XWPFParagraph xwpfParagraph : paragraphList) {
            CTP ctp = xwpfParagraph.getCTP();

            for (int dwI = 0; dwI < ctp.sizeOfBookmarkStartArray(); dwI++) {
                CTBookmark bookmark = ctp.getBookmarkStartArray(dwI);
                if (dataMap.containsKey(bookmark.getName())) {

                    XWPFRun run = xwpfParagraph.createRun();
                    run.setText(dataMap.get(bookmark.getName()));

                    Node firstNode = bookmark.getDomNode();
                    Node nextNode = firstNode.getNextSibling();
                    while (nextNode != null) {
                        // 循环查找结束符
                        String nodeName = nextNode.getNodeName();
                        if (nodeName.equals(BOOKMARK_END_TAG)) {
                            break;
                        }

                        // 删除中间的非结束节点,即删除原书签内容
                        Node delNode = nextNode;
                        nextNode = nextNode.getNextSibling();

                        ctp.getDomNode().removeChild(delNode);
                    }

                    if (nextNode == null) {
                        // 始终找不到结束标识的,就在书签前面添加
                        ctp.getDomNode().insertBefore(run.getCTR().getDomNode(), firstNode);
                    } else {
                        // 找到结束符,将新内容添加到结束符之前,即内容写入bookmark中间
                        ctp.getDomNode().insertBefore(run.getCTR().getDomNode(), nextNode);
                    }
                }
            }
        }

        document.write(outputStream);
        document.close();
    }


    /**
     * doc 文件中书签的替换
     *
     * @param inputStream
     * @param outputStream
     * @param dataMap
     * @throws IOException
     */
    public static void replaceBookmarksByDoc(InputStream inputStream, OutputStream outputStream,
                                             Map<String, String> dataMap) throws IOException {
        HWPFDocument document = new HWPFDocument(inputStream);
        Bookmarks bookmarks = document.getBookmarks();
        for (int dwI = 0; dwI < bookmarks.getBookmarksCount(); dwI++) {
            Bookmark bookmark = bookmarks.getBookmark(dwI);
            if (dataMap.containsKey(bookmark.getName())) {
                Range range = new Range(bookmark.getStart(), bookmark.getEnd(), document);
                range.replaceText(dataMap.get(bookmark.getName()), false);
            }
        }
        document.write(outputStream);
    }
}

(END)