生成文章标题有两种方式,一种是自定义标题,还有一种是使用现有样式来生成,现在网上大部分是第一种方式,但这种方式有很多弊端,我们先来看第一种

第一步,引入依赖

<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi</artifactId>
    <version>3.9</version>
</dependency>

<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi-ooxml</artifactId>
    <version>3.9</version>
</dependency>

代码如下:

/**
 * 自定义样式方式写word,参考statckoverflow的源码
 *
 * @throws IOException
 */
@Test
public void writeSimpleDocxFile() throws IOException {
    XWPFDocument docxDocument = new XWPFDocument();
    // 自定义标题1和标题2
    addCustomHeadingStyle(docxDocument, "标题 1", 1);
    addCustomHeadingStyle(docxDocument, "标题 2", 2);

    // 标题1
    XWPFParagraph paragraph = docxDocument.createParagraph();
    XWPFRun run = paragraph.createRun();
    run.setText("标题 1");
    run.setFontFamily("宋体");
    run.setBold(true);
    run.setFontSize(22);
    paragraph.setStyle("标题 1");

    // 标题2
    XWPFParagraph paragraph2 = docxDocument.createParagraph();
    XWPFRun run2 = paragraph2.createRun();
    run2.setText("标题 2");
    run2.setFontFamily("宋体");
    run2.setBold(true);
    run2.setFontSize(18);
    paragraph2.setStyle("标题 2");

    // 正文
    XWPFParagraph paragraphX = docxDocument.createParagraph();
    XWPFRun runX = paragraphX.createRun();
    runX.setText("正文");

    // word写入到文件
    FileOutputStream fos = new FileOutputStream("D:/myDoc.docx");
    docxDocument.write(fos);
    fos.close();
}

/**
 * 增加自定义标题样式。这里用的是stackoverflow的源码
 *
 * @param docxDocument 目标文档
 * @param strStyleId   样式名称
 * @param headingLevel 样式级别
 */
private static void addCustomHeadingStyle(XWPFDocument docxDocument, String strStyleId, int headingLevel) {
    CTStyle ctStyle = CTStyle.Factory.newInstance();
    ctStyle.setStyleId(strStyleId);
    CTString styleName = CTString.Factory.newInstance();
    styleName.setVal(strStyleId);
    ctStyle.setName(styleName);
    CTDecimalNumber indentNumber = CTDecimalNumber.Factory.newInstance();
    indentNumber.setVal(BigInteger.valueOf(headingLevel));
    // lower number > style is more prominent in the formats bar
    ctStyle.setUiPriority(indentNumber);

    CTOnOff onoffnull = CTOnOff.Factory.newInstance();
    ctStyle.setUnhideWhenUsed(onoffnull);
    // style shows up in the formats bar
    ctStyle.setQFormat(onoffnull);
    // style defines a heading of the given level
    CTPPr ppr = CTPPr.Factory.newInstance();
    ppr.setOutlineLvl(indentNumber);
    ctStyle.setPPr(ppr);
    XWPFStyle style = new XWPFStyle(ctStyle);
    // is a null op if already defined
    XWPFStyles styles = docxDocument.createStyles();
    style.setType(STStyleType.PARAGRAPH);
    styles.addStyle(style);
}

 生成的word文档WPS打开如下,发现目录栏新增了两个自定义目录,这样就没法用内置的标题1,2,3了否则层级会乱掉

XWPFTemplate 设置标题_java

 再看word打开的效果,发现虽然用了内置标题,但是内置标题的样式被覆盖了

XWPFTemplate 设置标题_java_02

 所以第一种方案我们不会采用,问题太多,接下去看方案二。

首先新建一个format.docx,内容空白的也没事,我们所有的标题样式都取自这个文档

/**
 * word整体样式
 */
private static CTStyles wordStyles = null;

/**
 * Word整体样式
 */
static {
    XWPFDocument template;
    try {
        // 读取模板文档
        template = new XWPFDocument(new FileInputStream("D:/format.docx"));
        // 获得模板文档的整体样式
        wordStyles = template.getStyle();
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    } catch (XmlException e) {
        e.printStackTrace();
    }
}

// 模板方式实现
@Test
public void formatDoc() throws IOException {
    // 新建的word文档对象
    XWPFDocument doc = new XWPFDocument();
    // 获取新建文档对象的样式
    XWPFStyles newStyles = doc.createStyles();
    // 关键行// 修改设置文档样式为静态块中读取到的样式
    newStyles.setStyles(wordStyles);

    // 开始内容输入
    // 标题1,1级大纲
    XWPFParagraph para1 = doc.createParagraph();
    // 关键行// 1级大纲
    para1.setStyle("2");
    XWPFRun run1 = para1.createRun();
    // 标题内容
    run1.setText("标题 1");

    // 标题2
    XWPFParagraph para2 = doc.createParagraph();
    // 关键行// 2级大纲
    para2.setStyle("3");
    XWPFRun run2 = para2.createRun();
    // 标题内容
    run2.setText("标题 2");

    // 正文
    XWPFParagraph paraX = doc.createParagraph();
    XWPFRun runX = paraX.createRun();
    // 正文内容
    runX.setText("正文");

    // word写入到文件
    FileOutputStream fos = new FileOutputStream("D:/myDoc.docx");
    doc.write(fos);
    fos.close();
}

来看关键代码newStyles.setStyles(wordStyles);这行代码的意思是使用format.docx的样式

再来看一个关键代码para1.setStyle("2");这行代码的意思是设置为标题1样式,关键是2是什么意思呢?2其实就是styleId,如何查看format.docx的styleId呢?

我们打开format.docx,另存为,保存成xml格式

XWPFTemplate 设置标题_java_03

打开保存的xml,找styleId

XWPFTemplate 设置标题_xml_04

可以看到heading 1对应的styleId是1,heading 2对应的styleId是2,heading 1和heading2又是什么鬼?

其实就是对应的目录栏,heading 1是正文,heading 2是标题1

XWPFTemplate 设置标题_自定义_05

 那如果修改标题1的样式呢?那就很简单了,右击标题1,点击修改样式就行了

XWPFTemplate 设置标题_poi_06