下载最新jar包:http://poi.apache.org/download.html 以及API

1.写入word

1.1 直接通过XWPFDocument生成

在使用XWPFDocument写docx文件时不需要像使用HWPFDocument写doc文件那样必须从一个doc文件开始,我们可以直接new一个空的XWPFDocument,之后再往这个XWPFDocument里面填充内容,然后再把它写入到对应的输出流中。下面是使用XWPFDocument生成docx文件的示例代码:

  1. public class XwpfTest {
  2.  
  3. /**
  4. * 基本的写操作
  5. * @throws Exception
  6. */
  7. @Test
  8. public void testSimpleWrite() throws Exception {
  9. //新建一个文档
  10. XWPFDocument doc = new XWPFDocument();
  11. //创建一个段落
  12. XWPFParagraph para = doc.createParagraph();
  13.  
  14. //一个XWPFRun代表具有相同属性的一个区域。
  15. XWPFRun run = para.createRun();
  16. run.setBold(true); //加粗
  17. run.setText("加粗的内容");
  18. run = para.createRun();
  19. run.setColor("FF0000");
  20. run.setText("红色的字。");
  21. OutputStream os = new FileOutputStream("D:\\simpleWrite.docx");
  22. //把doc输出到输出流
  23. doc.write(os);
  24. this.close(os);
  25. }
  26.  
  27. /***
  28. * 写一个表格
  29. * @throws Exception
  30. */
  31. @Test
  32. public void testWriteTable() throws Exception {
  33. XWPFDocument doc = new XWPFDocument();
  34. //创建一个5行5列的表格
  35. XWPFTable table = doc.createTable(5, 5);
  36. //这里增加的列原本初始化创建的那5行在通过getTableCells()方法获取时获取不到,但通过row新增的就可以。
  37. // table.addNewCol(); //给表格增加一列,变成6列
  38. table.createRow(); //给表格新增一行,变成6行
  39. List<XWPFTableRow> rows = table.getRows();
  40. //表格属性
  41. CTTblPr tablePr = table.getCTTbl().addNewTblPr();
  42. //表格宽度
  43. CTTblWidth width = tablePr.addNewTblW();
  44. width.setW(BigInteger.valueOf(8000));
  45. XWPFTableRow row;
  46. List<XWPFTableCell> cells;
  47. XWPFTableCell cell;
  48. int rowSize = rows.size();
  49. int cellSize;
  50. for (int i=0; i<rowSize; i++) {
  51. row = rows.get(i);
  52. //新增单元格
  53. row.addNewTableCell();
  54. //设置行的高度
  55. row.setHeight(500);
  56. //行属性
  57. // CTTrPr rowPr = row.getCtRow().addNewTrPr();
  58. //这种方式是可以获取到新增的cell的。
  59. // List<CTTc> list = row.getCtRow().getTcList();
  60. cells = row.getTableCells();
  61. cellSize = cells.size();
  62. for (int j=0; j<cellSize; j++) {
  63. cell = cells.get(j);
  64. if ((i+j)%2==0) {
  65. //设置单元格的颜色
  66. cell.setColor("ff0000"); //红色
  67. } else {
  68. cell.setColor("0000ff"); //蓝色
  69. }
  70. //单元格属性
  71. CTTcPr cellPr = cell.getCTTc().addNewTcPr();
  72. cellPr.addNewVAlign().setVal(STVerticalJc.CENTER);
  73. if (j == 3) {
  74. //设置宽度
  75. cellPr.addNewTcW().setW(BigInteger.valueOf(3000));
  76. }
  77. cell.setText(i + ", " + j);
  78. }
  79. }
  80. //文件不存在时会自动创建
  81. OutputStream os = new FileOutputStream("D:\\table.docx");
  82. //写入文件
  83. doc.write(os);
  84. this.close(os);
  85. }
  86.  
  87. /**
  88. * 关闭输出流
  89. * @param os
  90. */
  91. private void close(OutputStream os) {
  92. if (os != null) {
  93. try {
  94. os.close();
  95. } catch (IOException e) {
  96. e.printStackTrace();
  97. }
  98. }
  99. }
  100.  
  101. }

1.2 以docx文件作为模板

当然,我们也可以像写doc文件那样,先以一个docx文件作为模板,然后建立基于该docx文件的XWPFDocument对象,再把里面一些变化的信息在运行时进行替换,之后将XWPFDocument进行输出就可以了。所不同的是XWPFDocument中没有像HWPFDocument中那样的Range可以用来直接替换内容。而且底层的XWPFParagraph和XWPFRun也不支持直接将文本进行替换。倒是XWPFRun提供了一个设置文本的方法,不过新的文本不会替换旧的文本,而是会追加到原来的文本之后。现在的一个做法是先找出含有需要替换的变量的XWPFRun,然后将其移除,之后在原来的位置新增一个XWPFRun,其对应的文本是替换变量之后的文本。不过你设置的那个的变量的位置不一定就在一个XWPFRun里面,它有可能会被拆分到两个甚至更多的XWPFRun中,所以不是很有必要的话还是不推荐使用这种方式。

假设我们有一个docx文件,其内容是这样的:

之后我们以该文件作为模板,利用相关数据把里面的变量进行替换,然后把替换后的文档输出到另一个docx文件中。具体做法如下:

  1. public class XwpfTest {
  2.  
  3. /**
  4. * 用一个docx文档作为模板,然后替换其中的内容,再写入目标文档中。
  5. * @throws Exception
  6. */
  7. @Test
  8. public void testTemplateWrite() throws Exception {
  9. Map<String, Object> params = new HashMap<String, Object>();
  10. params.put("reportDate", "2014-02-28");
  11. params.put("appleAmt", "100.00");
  12. params.put("bananaAmt", "200.00");
  13. params.put("totalAmt", "300.00");
  14. String filePath = "D:\\word\\template.docx";
  15. InputStream is = new FileInputStream(filePath);
  16. XWPFDocument doc = new XWPFDocument(is);
  17. //替换段落里面的变量
  18. this.replaceInPara(doc, params);
  19. //替换表格里面的变量
  20. this.replaceInTable(doc, params);
  21. OutputStream os = new FileOutputStream("D:\\word\\write.docx");
  22. doc.write(os);
  23. this.close(os);
  24. this.close(is);
  25. }
  26.  
  27. /**
  28. * 替换段落里面的变量
  29. * @param doc 要替换的文档
  30. * @param params 参数
  31. */
  32. private void replaceInPara(XWPFDocument doc, Map<String, Object> params) {
  33. Iterator<XWPFParagraph> iterator = doc.getParagraphsIterator();
  34. XWPFParagraph para;
  35. while (iterator.hasNext()) {
  36. para = iterator.next();
  37. this.replaceInPara(para, params);
  38. }
  39. }
  40.  
  41. /**
  42. * 替换段落里面的变量
  43. * @param para 要替换的段落
  44. * @param params 参数
  45. */
  46. private void replaceInPara(XWPFParagraph para, Map<String, Object> params) {
  47. List<XWPFRun> runs;
  48. Matcher matcher;
  49. if (this.matcher(para.getParagraphText()).find()) {
  50. runs = para.getRuns();
  51. for (int i=0; i<runs.size(); i++) {
  52. XWPFRun run = runs.get(i);
  53. String runText = run.toString();
  54. matcher = this.matcher(runText);
  55. if (matcher.find()) {
  56. while ((matcher = this.matcher(runText)).find()) {
  57. runText = matcher.replaceFirst(String.valueOf(params.get(matcher.group(1))));
  58. }
  59. //直接调用XWPFRun的setText()方法设置文本时,在底层会重新创建一个XWPFRun,把文本附加在当前文本后面,
  60. //所以我们不能直接设值,需要先删除当前run,然后再自己手动插入一个新的run。
  61. para.removeRun(i);
  62. para.insertNewRun(i).setText(runText);
  63. }
  64. }
  65. }
  66. }
  67.  
  68. /**
  69. * 替换表格里面的变量
  70. * @param doc 要替换的文档
  71. * @param params 参数
  72. */
  73. private void replaceInTable(XWPFDocument doc, Map<String, Object> params) {
  74. Iterator<XWPFTable> iterator = doc.getTablesIterator();
  75. XWPFTable table;
  76. List<XWPFTableRow> rows;
  77. List<XWPFTableCell> cells;
  78. List<XWPFParagraph> paras;
  79. while (iterator.hasNext()) {
  80. table = iterator.next();
  81. rows = table.getRows();
  82. for (XWPFTableRow row : rows) {
  83. cells = row.getTableCells();
  84. for (XWPFTableCell cell : cells) {
  85. paras = cell.getParagraphs();
  86. for (XWPFParagraph para : paras) {
  87. this.replaceInPara(para, params);
  88. }
  89. }
  90. }
  91. }
  92. }
  93.  
  94. /**
  95. * 正则匹配字符串
  96. * @param str
  97. * @return
  98. */
  99. private Matcher matcher(String str) {
  100. Pattern pattern = Pattern.compile("\\$\\{(.+?)\\}", Pattern.CASE_INSENSITIVE);
  101. Matcher matcher = pattern.matcher(str);
  102. return matcher;
  103. }
  104.  
  105. /**
  106. * 关闭输入流
  107. * @param is
  108. */
  109. private void close(InputStream is) {
  110. if (is != null) {
  111. try {
  112. is.close();
  113. } catch (IOException e) {
  114. e.printStackTrace();
  115. }
  116. }
  117. }
  118.  
  119. /**
  120. * 关闭输出流
  121. * @param os
  122. */
  123. private void close(OutputStream os) {
  124. if (os != null) {
  125. try {
  126. os.close();
  127. } catch (IOException e) {
  128. e.printStackTrace();
  129. }
  130. }
  131. }
  132.  
  133. }

经过上面的代码所示的过程处理后,我们替换变量后新输出来的docx文件的内容是这样的:

POI写入word docx 07 的两种方法的更多相关文章

  1. Python 使用 xlwings 往 excel中写入一列数据的两种方法

    1.准备一个二维列表,然后再range后面不指定任何选项,可以输出该二维列表中数据在一列中显示,如下代码: # -*- coding:utf-8 -*- import xlwings as xw li ...

  2. POI读word docx 07 文件的两种方法

    POI在读写word docx文件时是通过xwpf模块来进行的,其核心是XWPFDocument.一个XWPFDocument代表一个docx文档,其可以用来读docx文档,也可以用来写docx文档. ...

  3. 前端调用后台接口下载word文档的两种方法

    1传统的ajax虽然能提交到后台,但是返回的数据被解析成json,html,text等字符串,无法响应浏览器下载.就算使用bob模拟下载,数据量大时也不方便 废话不多说:上代码(此处是Layui监听提 ...

  4. 清除SQLServer日志的两种方法

    日志文件满而造成SQL数据库无法写入文件时,可用两种方法:一种方法:清空日志.1.打开查询分析器,输入命令DUMP TRANSACTION 数据库名 WITH NO_LOG2.再打开企业管理器--右键 ...

  5. POI读写Word docx文件

    使用POI读写word docx文件 目录 1     读docx文件 1.1     通过XWPFWordExtractor读 1.2     通过XWPFDocument读 2     写docx ...

  6. 代码操作Word时,目录自动更新的两种方法

    最近的项目中有一个功能点为:根据分析数据库并生成报告.不过不是大数据.数据挖掘之类,报告的内容.组织方式都是事先固定下来的.实现的方式为,在普通word文档中插入书签制成模板,然后程序使用OpenXM ...

  7. Spark Streaming中空batches处理的两种方法(转)

    原文链接:Spark Streaming中空batches处理的两种方法 Spark Streaming是近实时(near real time)的小批处理系统.对给定的时间间隔(interval),S ...

  8. windows下获取IP地址的两种方法

    windows下获取IP地址的两种方法: 一种可以获取IPv4和IPv6,但是需要WSAStartup: 一种只能取到IPv4,但是不需要WSAStartup: 如下: 方法一:(可以获取IPv4和I ...

  9. [转载]C#读写txt文件的两种方法介绍

    C#读写txt文件的两种方法介绍 by 大龙哥 1.添加命名空间 System.IO; System.Text; 2.文件的读取 (1).使用FileStream类进行文件的读取,并将它转换成char ...

随机推荐

  1. JAG Practice Contest for ACM-ICPC Asia Regional 2016 C题【贪心】

    camp给出的题解: 题解:贪心,先算出最小需要的长度.然后从左到右依次确定每一位.复杂度O(n)O(n) 长度为 2n2n 的串可以构造出需要 [0,1+3+...+2n-1][0,1+3+...+ ...

  2. NGUI研究院之UISprite和UITexture浅谈

    NGUI的三大组件,UILabel.UISprite.UITexture,它们三个同时都继承UIWidget.先回到一个很郁闷的话题上,到底是优化DrawCall还是优化内存. UISprite : ...

  3. golang开发:(二)安装 Golang

    当然了我们的安装都是在vagrant里面安装,vagrant ssh.不用虚拟机了,本机安装当然也可以. Mac 安装 访问下载地址,下载 go1.12.5.darwin-amd64.pkg,双击下载 ...

  4. XHTML学习笔记 part1

    XHTML: 可扩展超文本标记语言 HTML语言最开始是用来描述文档的结构,如标题,段落等标记,后来HTML有增加了一些控制字体,对齐等方面的标记和属性,这样做的结果是HTML既可以用来描述文档的结构 ...

  5. Promise对象深入理解

     目录 基本用法 返回另一个 Promise 实例 Promise.prototypeof.then Promise.prototype.catch Promise.prototype.finally ...

  6. 【NOIP模拟】序列

    [问题描述] 一个序列被称为有趣的序列是它的所有的子串拥有一个唯一的整数(这个整数在整个序列中只出现过一次).给你一个序列的整数, 问你它是否是有趣的. [输入格式] 第一行 T, 表示数据组数.接下 ...

  7. ADO学途 five day 连接数据库

    用一个程序的目的就是为了方便对数据进行操作,没有数据的支持,程 序就成了一个空壳子.一般我们常用的数据库有三种mysql, SQL server, Oracle. C#中常用的就是SQL server ...

  8. IIS7文件无法下载问题处理

    使用IIS建立了静态站点,内部放置了一些文件供内部局域网下载使用,但deb等文件格式无法下载. 解决办法: 1.在IIS管理器中点击站点,选择右侧的MIME类型. 2.在MIME类型中添加需要下载文件 ...

  9. dubbo-spring

    一.需求 某个电商系统,订单服务需要调用用户服务获取某个用户的所有地址:创建两个服务模块进行测试 测试预期结果:订单服务web模块在A服务器,用户服务模块在B服务器,A可以远程调用B的功能. 二.工程 ...

  10. 535 Encode and Decode TinyURL 编码和解码精简URL地址

    详见:https://leetcode.com/problems/encode-and-decode-tinyurl/description/ C++: class Solution { public ...