java⽣成word⽂档的多种⽅法(三)
这篇⽂章给⼤家介绍java  通过POI技术⽣成word ,这种实现⽅式项⽬中也很常⽤,⽐如⽣成常⽤表格,报表等。废话少说,下⾯就给⼤家详细介绍实现原理!
A、word⽂档正⽂段落概述:
⼀个word⽂档包含多个段落,⼀个段落包含多个Runs,⼀个Runs包含多个Run,Run是⽂档的最⼩单位
获取所有段落:List<XWPFParagraph> paragraphs = Paragraphs();
获取⼀个段落中的所有Runs:List<XWPFRun> xwpfRuns = Runs();
获取⼀个Runs中的⼀个Run:XWPFRun run = (index);
B、word⽂档正⽂表格概述:
⼀个⽂档可包含多个表格,每个表格可以包含多⾏,每⾏可以包含多列(格),每⼀格的内容相当于⼀个完整的⽂档
获取所有表格:List<XWPFTable> xwpfTables = Tables();
获取⼀个表格中的所有⾏:List<XWPFTableRow> xwpfTableRows = Rows();
获取⼀⾏中的所有列:List<XWPFTableCell> xwpfTableCells = TableCells();
获取⼀格⾥的内容:List<XWPFParagraph> paragraphs = Paragraphs();
A,B介绍有利于⼤家理解POI实现原理,所需jar包如图:这⼏个jar包⾜够
⼀、⾸先、该word模板是直接⽤word⽂档做模板,如下图:模板⾥⾯的参数⼤家可以随意定义,可以⽤${name},也可不⽤,直接  name
⼆、咱们就利⽤这个模板做测试,⽣成⽂档和⽂档⾥⾯的表格,测试成功直接可以使⽤,直接上代码:如下
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import org.apache.poi.POIXMLDocument;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.apache.poi.xwpf.usermodel.XWPFParagraph;
import org.apache.poi.xwpf.usermodel.XWPFRun;
import org.apache.poi.xwpf.usermodel.XWPFTable;
import org.apache.poi.xwpf.usermodel.XWPFTableCell;
import org.apache.poi.xwpf.usermodel.XWPFTableRow;
/**
* 通过word模板⽣成新的word⼯具类
*
*/
public class BokeWordUtils {
/**
* 根据模板⽣成新word⽂档
* 判断表格是需要替换还是需要插⼊,判断逻辑有$为替换,表格⽆$为插⼊
* @param inputUrl 模板存放地址
* @param outPutUrl 新⽂档存放地址
* @param textMap 需要替换的信息集合
* @param tableList 需要插⼊的表格信息集合
*/
public static boolean changWord(String inputUrl, String outputUrl,
Map<String, String> textMap, List<String[]> tableList) {
//模板转换默认成功
boolean changeFlag = true;
try {
//获取docx解析对象
XWPFDocument document = new XWPFDocument(POIXMLDocument.openPackage(inputUrl));            //解析替换⽂本段落对象
BokeWordUtils.changeText(document, textMap);
//解析替换表格对象
BokeWordUtils.changeTable(document, textMap, tableList);
//⽣成新的word
File file = new File(outputUrl);
FileOutputStream stream = new FileOutputStream(file);
document.write(stream);
stream.close();
System.out.println("成功⽣成!");
} catch (IOException e) {
e.printStackTrace();
changeFlag = false;
}
return changeFlag;
}
/**
* 替换段落⽂本
* @param document docx解析对象
* @param textMap 需要替换的信息集合
*/
public static void changeText(XWPFDocument document, Map<String, String> textMap){
/
/获取段落集合
List<XWPFParagraph> paragraphs = Paragraphs();
for (XWPFParagraph paragraph : paragraphs) {
//判断此段落时候需要进⾏替换
String text = Text();
if(checkText(text)){
List<XWPFRun> runs = Runs();
for (XWPFRun run : runs) {
//替换模板原来位置
// run.setText(String(), textMap),0);
String textValue = String(), textMap);
run.setText(textValue,0);
}
}
}
}
/**
* 替换表格对象⽅法
* @param document docx解析对象
* @param textMap 需要替换的信息集合
* @param tableList 需要插⼊的表格信息集合
*/
public static void changeTable(XWPFDocument document, Map<String, String> textMap,            List<String[]> tableList){
//获取表格对象集合
List<XWPFTable> tables = Tables();
for (int i = 0; i < tables.size(); i++) {
//只处理⾏数⼤于等于2的表格,且不循环表头
XWPFTable table = (i);
Rows().size()>1){
//判断表格是需要替换还是需要插⼊,判断逻辑有$为替换,表格⽆$为插⼊
if(Text())){
List<XWPFTableRow> rows = Rows();
//遍历表格,并替换模板
eachTable(rows, textMap);
}else{
//                  System.out.println("插⼊"+Text());
insertTable(table, tableList);
}
}
}
}
/**
* 遍历表格
* @param rows 表格⾏对象
* @param textMap 需要替换的信息集合
*/
public static void eachTable(List<XWPFTableRow> rows ,Map<String, String> textMap){        for (XWPFTableRow row : rows) {
List<XWPFTableCell> cells = TableCells();
for (XWPFTableCell cell : cells) {
//判断单元格是否需要替换
if(Text())){
List<XWPFParagraph> paragraphs = Paragraphs();
for (XWPFParagraph paragraph : paragraphs) {
List<XWPFRun> runs = Runs();
for (XWPFRun run : runs) {
run.setText(String(), textMap),0);
}
}
}
}
}
}
/**
* 为表格插⼊数据,⾏数不够添加新⾏
* 为表格插⼊数据,⾏数不够添加新⾏
* @param table 需要插⼊数据的表格
* @param tableList 插⼊数据集合
*/
public static void insertTable(XWPFTable table, List<String[]> tableList){
//创建⾏,根据需要插⼊的数据添加新⾏,不处理表头
for(int i = 1; i < tableList.size(); i++){
XWPFTableRow row =ateRow();
}
//遍历表格插⼊数据
List<XWPFTableRow> rows = Rows();
for(int i = 1; i < rows.size(); i++){
XWPFTableRow newRow = Row(i);
List<XWPFTableCell> cells = TableCells();
for(int j = 0; j < cells.size(); j++){
XWPFTableCell cell = (j);
cell.(i-1)[j]);
}
}
}
/**
* 判断⽂本中时候包含$
java replace方法* @param text ⽂本
* @return 包含返回true,不包含返回false
*/
public static boolean checkText(String text){
boolean check  =  false;
if(text.indexOf("$")!= -1){
check = true;
}
return check;
}
/**
* 匹配传⼊信息集合与模板
* @param value 模板需要替换的区域
* @param textMap 传⼊信息集合
* @return 模板需要替换区域信息集合对应值
*/
public static String changeValue(String value, Map<String, String> textMap){        Set<Entry<String, String>> textSets = Set();
for (Entry<String, String> textSet : textSets) {
//匹配模板与替换值格式${key}
String key = "${"+Key()+"}";
if(value.indexOf(key)!= -1){
// value = Value();//全部参数替换
value = place(key, Value());//仅替换参数
}
}
//模板未匹配到区域替换为空
if(checkText(value)){
value = "";
}
return value;
}
public static void main(String[] args) {
//模板⽂件地址
String inputUrl = "F://testWord/test0.docx";
//新⽣产的模板⽂件
String outputUrl = "F://testWord/testBoke.docx";
Map<String, String> testMap = new HashMap<String, String>();
testMap.put("name", "⼩明");
testMap.put("sex", "男");
testMap.put("nling", "18");
testMap.put("address", "北京市");
testMap.put("address", "北京市");
testMap.put("neirong", "好的内容");
testMap.put("address", "软件园");
testMap.put("xuehao", "88888888");
List<String[]> testList = new ArrayList<String[]>();
testList.add(new String[]{"1","1mm","1开开","1uu"});
testList.add(new String[]{"2","2密码","2B","基⾦C"});
testList.add(new String[]{"3","3看看","3B","基看C"});
testList.add(new String[]{"4","4累了","4B","4谁说的"});
BokeWordUtils.changWord(inputUrl, outputUrl, testMap, testList);
}
}
三、次代码是我在⽹上查询出来的,经过修改优化,成功使⽤。改模板代码参数设置为${name},类似EL表达式。同时这种⽅式能很好地避免Runs分段混乱问题(分段混乱原因:是${name}是⼿打上去的,如果先在记事本输⼊,再复制粘贴进docx则不会出现分段问题)。代码⾮常详细,⼤家有不明⽩⼤地⽅及时留⾔
四、下篇⽂章给⼤家介绍word⽂档上有图⽚⽣成的问题,敬请期待

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。