【word多文档docx合并】

word多文档docx合并

一、背景

项目起源

需要将多个独立的Word文档合并成一个完整的文档。需要保留各个word的页面方向、纸张大小、页边距等复杂的页面布局,尽量保证word表格不变形,不失真

二、核心原理

  • word本身可以用压缩包工具解压
  • 核心内容在于解压后的word目录的document.xml和style.xml
  • 结合docx4j获取上述两个xml的完整内容并转为java对象
  • 正确处理Word文档中两种节设置位置
    • bodysectPr(文档结束时的页面设置)
    • contentpPrsectPr(段落级别的分节符)
    • 保持每个文档原有的页面方向、纸张大小、页边距设置

三、技术实现要点

java 复制代码
// 预处理原始文档
List<String> processedDocPathList = PreprocessDocumentUtil.preprocessDocumentList(batchDocPaths);

// 加载当前批次数据
List<WordprocessingMLPackage> docList = WordProcessingUtils.loadDocList(processedDocPathList);

WordprocessingMLPackage resultDoc;
if (baseDoc == null) {
    // 如果没有基础文档,使用第一个文档作为基础
    resultDoc = docList.get(0);
} else {
    // 如果已有基础文档,将其作为第一个文档
    docList.add(0, baseDoc);
    resultDoc = baseDoc;
}

// 合并样式,以第一个文档的样式为基准
StyleReMapperUtil.mergeStyles(docList);

// 映射编号(避免列表编号混乱)
NumberingMapperUtil.mapNumbering(docList);

// 复制图片、表格等资源(处理关系)
ResourceCopierUtil.copyImages(docList);

// 保存两个文档的格式信息(暂时保留但不处理表格边框)
Map<String, String> formatProperties = TableFormatPreserverUtil.saveDocumentFormat(docList);

// 在合并前应用默认字体大小
FontMapperUtil.applyDocListDefaultFontSizesBeforeMerge(docList, formatProperties);

// 获取基础文档的主要部分
MainDocumentPart main1 = resultDoc.getMainDocumentPart();

// 移除文档网格设置
WordProcessingUtils.removeDocumentGridSettingsList(docList);

// 将合并doc 的所有内容追加到 doc1
WordProcessingUtils.addDocListToBase(main1, docList);

四、使用方法

基本使用

java 复制代码
DocxMerger merger = new DocxMerger();
List<File> docFiles = Arrays.asList(
    new File("document1.docx"),
    new File("document2.docx")
);
File outputFile = new File("merged_result.docx");
merger.mergeDocuments(docFiles, outputFile);

GitHub仓库地址:https://github.com/LLBlood/docx-merge-advanced.git

待解决的问题

  • 页眉页脚
  • 目录
  • 页码
  • 等等杂七杂八的数据
相关推荐
爱学习的小可爱卢11 分钟前
编程语言30年:从Java到Rust的进化史
java·开发语言·rust
一 乐13 分钟前
校园社区系统|基于java+vue的校园悬赏任务平台系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot
就叫飞六吧21 分钟前
考古spring.xml注册bean无法扫描目录问题
xml·java·spring
markuszhang24 分钟前
G1 垃圾回收器学习
java·学习
CoderYanger24 分钟前
动态规划算法-子序列问题(数组中不连续的一段):30.最长数对链
java·算法·leetcode·动态规划·1024程序员节
星月IWJ25 分钟前
领域驱动设计学习
java·学习·设计模式
毕设源码-郭学长27 分钟前
【开题答辩全过程】以 基于ssm的企业进销存管理系统的设计与实现为例,包含答辩的问题和答案
java
镜花水月linyi42 分钟前
1.5w字ReentrantLock 深度解析
java·后端·程序员
程序员西西1 小时前
Spring Boot3 分页操作全解析:从基础到实战
java·后端·程序员
Boop_wu1 小时前
[Java EE] 网络原理(1)
java·网络·java-ee