?? 近段時(shí)間因?yàn)轫?xiàng)目比較忙所以很少來(lái)javaEye溜轉(zhuǎn),現(xiàn)在終于忙完了。這期間做了一個(gè)word模板填充的功能,覺(jué)得應(yīng)該放上來(lái)保存下。
?? 問(wèn)題的緣由是這樣的,項(xiàng)目里面有個(gè)功能是下載記錄客戶明細(xì)的一個(gè)word,之前的做法是將整個(gè)word以流的形式寫(xiě)在類里面,然后將值填充到word流里面需要填充地方,最后使用sturts2的文件機(jī)制將word流提供給客戶另存為word文件。這里面的問(wèn)題就是項(xiàng)目里面每增加一個(gè)模板就要去在類里面寫(xiě)一次word流,而且需要組裝的數(shù)據(jù)里包含了很多的迭代等操作,所以這塊的工作量很大,而且很不好維護(hù)。于是老大讓我將這一塊的word不要在類里面直接去處理word流。
??? 最后我使用的方法是將需要填充的word做成一個(gè)通用的模板,里面需要填充數(shù)據(jù)的地方使用特定的代號(hào)去取代,然后將這個(gè)這個(gè)word放到項(xiàng)目中?;镜乃枷胧钱?dāng)要用到這個(gè)模板的時(shí)候通過(guò)程序?qū)ord讀取成流的形式,然后在類里面將需要填充的數(shù)據(jù)組裝成一到一個(gè)HashMap里,鍵和通用模板的代號(hào)一致,值是將要填充的內(nèi)容。再寫(xiě)一個(gè)類來(lái)將HashMap里面的值替換掉word流里對(duì)應(yīng)的鍵。 最后將組裝后的流輸出。
??? 這個(gè)好像說(shuō)起來(lái)很拗口,也不好解釋 下面以圖文的形式來(lái)解釋把:
1.先拿到一個(gè)他們現(xiàn)在在用的word,
將word里面需要填充的地方使用[替代碼]來(lái)替代
2.在類里面進(jìn)行hashmap的組裝
public static void main(String[] args) throws Exception {
DocGenerator gen = new DocGenerator();
Map data = new HashMap();
data.put("[ORDER_TITLE]", "**科技集團(tuán)差旅管理服務(wù)公司酒店預(yù)訂單(酒店聯(lián))");
data.put("[ORDER_NO]", "HB080709101712996");
data.put("[OPTRID]", "IHZYF011");
//子項(xiàng)
List lstValue = new ArrayList();
Map subData = new HashMap();
subData.put("[CUST_NAME]", "張三");
subData.put("[ROOM_STYLE]", "單床房? 大床? 寬帶 免費(fèi)");
subData.put("[DATE1]", "2008-07-09抵? 2008-07-12離");
subData.put("[DATE2]", "2008-07-09 至 2008-07-12 每晚 111 元 (單早)");
subData.put("[DATE3]", "2008-07-09 至 2008-07-12 每晚 1.0 間");
lstValue.add(subData);
subData = new HashMap();
subData.put("[CUST_NAME]", "李四");
subData.put("[ROOM_STYLE]", "雙人床房? 大床? 寬帶 免費(fèi)");
subData.put("[DATE1]", "2008-07-09抵? 2008-07-12離");
subData.put("[DATE2]", "2008-07-09 至 2008-07-12 每晚 111 元 (單早)");
subData.put("[DATE3]", "2008-07-09 至 2008-07-12 每晚 1.0 間");
lstValue.add(subData);
data.put("CUSTS", lstValue);
gen.generatorDoc(data, "1");//調(diào)用處理方法
}
3.處理類書(shū)寫(xiě)
/**
* 模板生成類
* @author chm20081009
*
*/
public class DocGenerator {
/**
* 根據(jù)傳入的數(shù)據(jù)和模板ID生成
* @author chm
* @param data
* @param templetid
* @throws Exception
*/
@SuppressWarnings("unchecked")
public static String generatorDoc(Map data, String templetid) throws Exception {
String result = "";
// 根據(jù)templetid獲取模板
String templet = getTemplet(templetid);
//處理模板
templet =split(data,templet);
//去除沒(méi)有數(shù)據(jù)的定義變量
templet = templet.replaceAll("\\[\\w+]", "");
//saveDoc(templet);
return templet;
}
/**
* 拆分模板
* @author chm
* @param data
* @param templet
* @return
*/
private static String split(Map data,String templet){
//
Iterator iter = data.keySet().iterator();
//System.out.println("templet==="+templet);
// templet = "哈哈哈[ORDER_TITLE]測(cè)試數(shù)據(jù)";
while (iter.hasNext()) {
String key = iter.next().toString();
Object value = data.get(key);
//替換模板上的標(biāo)簽
templet = replace(key,value,templet);
}
return templet;
}
/**
* 負(fù)責(zé)替換
* @return
*/
private static String replace(String key , Object value ,String templet){
if (value instanceof String) {// 單值替換
templet = templet.replaceAll("\\" + key, value.toString());
} else if (value instanceof List) {// 列表
// 從模板中截取迭代定義部分,把整個(gè)模板分成3部分
String firstTemplet = templet.substring(0, templet
.indexOf("[ITERATE_BEGIN_" + key + "]"));
String middleTemplet = templet.substring(templet
.indexOf("[ITERATE_BEGIN_" + key + "]")
+ ("[ITERATE_BEGIN_" + key + "]").length(), templet
.indexOf("[ITERATE_END_" + key + "]"));
String lastTemplet = templet.substring(templet
.indexOf("[ITERATE_END_" + key + "]")
+ ("[ITERATE_END_" + key + "]").length());
List lstValue = (List) value;// 里面存放的是Map數(shù)據(jù)
StringBuffer middleResult = new StringBuffer(100);
/**
* 遞歸調(diào)用
*/
for (int i = 0; i < lstValue.size(); i++) {
Map subData = (Map) lstValue.get(i);
String middleData = middleTemplet;
middleData = split(subData,middleData);
middleResult.append(middleData);
}
//將中間替換的數(shù)據(jù)和拆分開(kāi)的前后半部分模板組裝起來(lái)
templet = firstTemplet + middleResult.toString() + lastTemplet;
}
return templet;
}
/**
* 根據(jù)模板ID獲取模板的內(nèi)容
* @param templetid
* @return
* @throws Exception
*/
private static String getTemplet(String templetid) throws Exception {
StringBuffer result = new StringBuffer(100);
/*
*可以根據(jù)模板ID先到cache里面進(jìn)行匹配,匹配到模板,直接從緩存中獲取
*/
// String fileName = getTempletName(templetid);//寫(xiě)死的文件路徑
String fileName = TempletCache.getTemplet(templetid);//從內(nèi)存根據(jù)id獲取模板完全路徑
BufferedReader in = null;
try {
// in = new BufferedReader(new InputStreamReader(new FileInputStream(
// fileName), "UTF-8"));
in = new BufferedReader(new InputStreamReader(TempletCache.class
.getClassLoader().getResourceAsStream(fileName), "UTF-8"));
String s = "";
while ((s = in.readLine()) != null) {
result.append(s);
result.append("\n");
}
} finally {
try {
if (in != null) {
in.close();
}
} catch (Exception e) {
}
}
return result.toString();
}
}
4.寫(xiě)了一個(gè)緩存類
class TempletCache {
private static final String CONFIG_FILE_NAME = "config.properties";
private static HashMap<String, String> hmTemplet = new HashMap<String, String>();
private static final Properties prop = new Properties();
private static TempletCache instance;
//讀取配置文件? 模板的路徑是存儲(chǔ)在config.properties
private static ResourceBundle prop1 = ResourceBundle.getBundle("config");
private TempletCache() {
}
public static synchronized TempletCache getInstance() throws Exception {
if (instance == null) {
instance = new TempletCache();
instance.loadConfig();
}
return instance;
}
/**
* 加載模板配置文件
*
* @return
* @throws Exception
*/
private static void loadConfig() throws Exception {
InputStream fis = null;
try {
fis = TempletCache.class.getClass().getClassLoader().getResourceAsStream(
CONFIG_FILE_NAME);
prop.load(fis);
} finally {
try {
if (fis != null) {
fis.close();
}
} catch (Exception e) {
}
}
}
/**
* 根據(jù)模板id獲取模板路徑
* 策略:從內(nèi)存中根據(jù)模板id獲得模板路徑,如果內(nèi)存中沒(méi)有此id則從配置文件中讀取該id讀出模板路徑同時(shí)將模板放入內(nèi)存中
* @param templetid
* @return
* @throws Exception
*/
public static String getTemplet(String templetid) throws Exception {
String templet = "";
if (hmTemplet.containsKey(templetid)) {
templet = hmTemplet.get(templetid);
} else {
templet = prop1.getString(templetid);
hmTemplet.put(templetid, templet);
}
return templet;
}
這里將模板和對(duì)應(yīng)的模板存放路徑放入內(nèi)存中,當(dāng)需要去生成一個(gè)模板的時(shí)候,根據(jù)已知模板編號(hào)(一個(gè)類型傳真模板是同一個(gè))去從內(nèi)存中去獲取模板存放的路徑,如果內(nèi)存中無(wú)該編號(hào),則到配置文件config.properties中根據(jù)模板ID去讀取對(duì)應(yīng)的模板路徑,同時(shí)將該路徑和模板編號(hào)存放在 hashma中
5.通過(guò)DocGenerator.java 里面的遞歸迭代將模板流中與DATA中key值相同的地方使用該key對(duì)應(yīng)的value替換
這里需要與前面的2做很好的銜接,2做的是組裝而這里做的就是解析替換,所有的key必須一致,特別是當(dāng)模板中有列表的地方
注:當(dāng)一個(gè)word用編輯器打開(kāi)的時(shí)候其代碼是很冗長(zhǎng)的,我沒(méi)又沒(méi)類似Dreamvaver來(lái)可視化編輯html一樣的工具來(lái)編輯word,這是一個(gè)頭痛的問(wèn)題,不過(guò)如果自己查看,word流也是有他的規(guī)律的,什么時(shí)候段落是什么時(shí)候表格結(jié)束什么時(shí)候換行,都有他的特定的標(biāo)識(shí)(類似html中<table></table>表示表格,<td></td>表示單元格一樣)
6.最終將替換后的流輸出保存成文件,就是我們最后的需要的word了
?? 問(wèn)題的緣由是這樣的,項(xiàng)目里面有個(gè)功能是下載記錄客戶明細(xì)的一個(gè)word,之前的做法是將整個(gè)word以流的形式寫(xiě)在類里面,然后將值填充到word流里面需要填充地方,最后使用sturts2的文件機(jī)制將word流提供給客戶另存為word文件。這里面的問(wèn)題就是項(xiàng)目里面每增加一個(gè)模板就要去在類里面寫(xiě)一次word流,而且需要組裝的數(shù)據(jù)里包含了很多的迭代等操作,所以這塊的工作量很大,而且很不好維護(hù)。于是老大讓我將這一塊的word不要在類里面直接去處理word流。
??? 最后我使用的方法是將需要填充的word做成一個(gè)通用的模板,里面需要填充數(shù)據(jù)的地方使用特定的代號(hào)去取代,然后將這個(gè)這個(gè)word放到項(xiàng)目中?;镜乃枷胧钱?dāng)要用到這個(gè)模板的時(shí)候通過(guò)程序?qū)ord讀取成流的形式,然后在類里面將需要填充的數(shù)據(jù)組裝成一到一個(gè)HashMap里,鍵和通用模板的代號(hào)一致,值是將要填充的內(nèi)容。再寫(xiě)一個(gè)類來(lái)將HashMap里面的值替換掉word流里對(duì)應(yīng)的鍵。 最后將組裝后的流輸出。
??? 這個(gè)好像說(shuō)起來(lái)很拗口,也不好解釋 下面以圖文的形式來(lái)解釋把:
1.先拿到一個(gè)他們現(xiàn)在在用的word,

將word里面需要填充的地方使用[替代碼]來(lái)替代

2.在類里面進(jìn)行hashmap的組裝
public static void main(String[] args) throws Exception {
DocGenerator gen = new DocGenerator();
Map data = new HashMap();
data.put("[ORDER_TITLE]", "**科技集團(tuán)差旅管理服務(wù)公司酒店預(yù)訂單(酒店聯(lián))");
data.put("[ORDER_NO]", "HB080709101712996");
data.put("[OPTRID]", "IHZYF011");
//子項(xiàng)
List lstValue = new ArrayList();
Map subData = new HashMap();
subData.put("[CUST_NAME]", "張三");
subData.put("[ROOM_STYLE]", "單床房? 大床? 寬帶 免費(fèi)");
subData.put("[DATE1]", "2008-07-09抵? 2008-07-12離");
subData.put("[DATE2]", "2008-07-09 至 2008-07-12 每晚 111 元 (單早)");
subData.put("[DATE3]", "2008-07-09 至 2008-07-12 每晚 1.0 間");
lstValue.add(subData);
subData = new HashMap();
subData.put("[CUST_NAME]", "李四");
subData.put("[ROOM_STYLE]", "雙人床房? 大床? 寬帶 免費(fèi)");
subData.put("[DATE1]", "2008-07-09抵? 2008-07-12離");
subData.put("[DATE2]", "2008-07-09 至 2008-07-12 每晚 111 元 (單早)");
subData.put("[DATE3]", "2008-07-09 至 2008-07-12 每晚 1.0 間");
lstValue.add(subData);
data.put("CUSTS", lstValue);
gen.generatorDoc(data, "1");//調(diào)用處理方法
}
3.處理類書(shū)寫(xiě)
/**
* 模板生成類
* @author chm20081009
*
*/
public class DocGenerator {
/**
* 根據(jù)傳入的數(shù)據(jù)和模板ID生成
* @author chm
* @param data
* @param templetid
* @throws Exception
*/
@SuppressWarnings("unchecked")
public static String generatorDoc(Map data, String templetid) throws Exception {
String result = "";
// 根據(jù)templetid獲取模板
String templet = getTemplet(templetid);
//處理模板
templet =split(data,templet);
//去除沒(méi)有數(shù)據(jù)的定義變量
templet = templet.replaceAll("\\[\\w+]", "");
//saveDoc(templet);
return templet;
}
/**
* 拆分模板
* @author chm
* @param data
* @param templet
* @return
*/
private static String split(Map data,String templet){
//
Iterator iter = data.keySet().iterator();
//System.out.println("templet==="+templet);
// templet = "哈哈哈[ORDER_TITLE]測(cè)試數(shù)據(jù)";
while (iter.hasNext()) {
String key = iter.next().toString();
Object value = data.get(key);
//替換模板上的標(biāo)簽
templet = replace(key,value,templet);
}
return templet;
}
/**
* 負(fù)責(zé)替換
* @return
*/
private static String replace(String key , Object value ,String templet){
if (value instanceof String) {// 單值替換
templet = templet.replaceAll("\\" + key, value.toString());
} else if (value instanceof List) {// 列表
// 從模板中截取迭代定義部分,把整個(gè)模板分成3部分
String firstTemplet = templet.substring(0, templet
.indexOf("[ITERATE_BEGIN_" + key + "]"));
String middleTemplet = templet.substring(templet
.indexOf("[ITERATE_BEGIN_" + key + "]")
+ ("[ITERATE_BEGIN_" + key + "]").length(), templet
.indexOf("[ITERATE_END_" + key + "]"));
String lastTemplet = templet.substring(templet
.indexOf("[ITERATE_END_" + key + "]")
+ ("[ITERATE_END_" + key + "]").length());
List lstValue = (List) value;// 里面存放的是Map數(shù)據(jù)
StringBuffer middleResult = new StringBuffer(100);
/**
* 遞歸調(diào)用
*/
for (int i = 0; i < lstValue.size(); i++) {
Map subData = (Map) lstValue.get(i);
String middleData = middleTemplet;
middleData = split(subData,middleData);
middleResult.append(middleData);
}
//將中間替換的數(shù)據(jù)和拆分開(kāi)的前后半部分模板組裝起來(lái)
templet = firstTemplet + middleResult.toString() + lastTemplet;
}
return templet;
}
/**
* 根據(jù)模板ID獲取模板的內(nèi)容
* @param templetid
* @return
* @throws Exception
*/
private static String getTemplet(String templetid) throws Exception {
StringBuffer result = new StringBuffer(100);
/*
*可以根據(jù)模板ID先到cache里面進(jìn)行匹配,匹配到模板,直接從緩存中獲取
*/
// String fileName = getTempletName(templetid);//寫(xiě)死的文件路徑
String fileName = TempletCache.getTemplet(templetid);//從內(nèi)存根據(jù)id獲取模板完全路徑
BufferedReader in = null;
try {
// in = new BufferedReader(new InputStreamReader(new FileInputStream(
// fileName), "UTF-8"));
in = new BufferedReader(new InputStreamReader(TempletCache.class
.getClassLoader().getResourceAsStream(fileName), "UTF-8"));
String s = "";
while ((s = in.readLine()) != null) {
result.append(s);
result.append("\n");
}
} finally {
try {
if (in != null) {
in.close();
}
} catch (Exception e) {
}
}
return result.toString();
}
}
4.寫(xiě)了一個(gè)緩存類
class TempletCache {
private static final String CONFIG_FILE_NAME = "config.properties";
private static HashMap<String, String> hmTemplet = new HashMap<String, String>();
private static final Properties prop = new Properties();
private static TempletCache instance;
//讀取配置文件? 模板的路徑是存儲(chǔ)在config.properties
private static ResourceBundle prop1 = ResourceBundle.getBundle("config");
private TempletCache() {
}
public static synchronized TempletCache getInstance() throws Exception {
if (instance == null) {
instance = new TempletCache();
instance.loadConfig();
}
return instance;
}
/**
* 加載模板配置文件
*
* @return
* @throws Exception
*/
private static void loadConfig() throws Exception {
InputStream fis = null;
try {
fis = TempletCache.class.getClass().getClassLoader().getResourceAsStream(
CONFIG_FILE_NAME);
prop.load(fis);
} finally {
try {
if (fis != null) {
fis.close();
}
} catch (Exception e) {
}
}
}
/**
* 根據(jù)模板id獲取模板路徑
* 策略:從內(nèi)存中根據(jù)模板id獲得模板路徑,如果內(nèi)存中沒(méi)有此id則從配置文件中讀取該id讀出模板路徑同時(shí)將模板放入內(nèi)存中
* @param templetid
* @return
* @throws Exception
*/
public static String getTemplet(String templetid) throws Exception {
String templet = "";
if (hmTemplet.containsKey(templetid)) {
templet = hmTemplet.get(templetid);
} else {
templet = prop1.getString(templetid);
hmTemplet.put(templetid, templet);
}
return templet;
}
這里將模板和對(duì)應(yīng)的模板存放路徑放入內(nèi)存中,當(dāng)需要去生成一個(gè)模板的時(shí)候,根據(jù)已知模板編號(hào)(一個(gè)類型傳真模板是同一個(gè))去從內(nèi)存中去獲取模板存放的路徑,如果內(nèi)存中無(wú)該編號(hào),則到配置文件config.properties中根據(jù)模板ID去讀取對(duì)應(yīng)的模板路徑,同時(shí)將該路徑和模板編號(hào)存放在 hashma中

5.通過(guò)DocGenerator.java 里面的遞歸迭代將模板流中與DATA中key值相同的地方使用該key對(duì)應(yīng)的value替換
這里需要與前面的2做很好的銜接,2做的是組裝而這里做的就是解析替換,所有的key必須一致,特別是當(dāng)模板中有列表的地方
注:當(dāng)一個(gè)word用編輯器打開(kāi)的時(shí)候其代碼是很冗長(zhǎng)的,我沒(méi)又沒(méi)類似Dreamvaver來(lái)可視化編輯html一樣的工具來(lái)編輯word,這是一個(gè)頭痛的問(wèn)題,不過(guò)如果自己查看,word流也是有他的規(guī)律的,什么時(shí)候段落是什么時(shí)候表格結(jié)束什么時(shí)候換行,都有他的特定的標(biāo)識(shí)(類似html中<table></table>表示表格,<td></td>表示單元格一樣)
6.最終將替換后的流輸出保存成文件,就是我們最后的需要的word了
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫(xiě)作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元
