word模板制作

本文介绍了一种使用Java实现Word文档模板动态填充数据的方法,通过定义特定占位符并利用HashMap进行数据组装,实现了灵活高效的Word文档生成流程。
部署运行你感兴趣的模型镜像
近段时间因为项目比较忙所以很少来javaEye溜转,现在终于忙完了。这期间做了一个word模板填充的功能,觉得应该放上来保存下。
问题的缘由是这样的,项目里面有个功能是下载记录客户明细的一个word,之前的做法是将整个word以流的形式写在类里面,然后将值填充到word流里面需要填充地方,最后使用sturts2的文件机制将word流提供给客户另存为word文件。这里面的问题就是项目里面每增加一个模板就要去在类里面写一次word流,而且需要组装的数据里包含了很多的迭代等操作,所以这块的工作量很大,而且很不好维护。于是老大让我将这一块的word不要在类里面直接去处理word流。
最后我使用的方法是将需要填充的word做成一个通用的模板,里面需要填充数据的地方使用特定的代号去取代,然后将这个这个word放到项目中。基本的思想是当要用到这个模板的时候通过程序将word读取成流的形式,然后在类里面将需要填充的数据组装成一到一个HashMap里,键和通用模板的代号一致,值是将要填充的内容。再写一个类来将HashMap里面的值替换掉word流里对应的键。 最后将组装后的流输出。
这个好像说起来很拗口,也不好解释 下面以图文的形式来解释把:
1.先拿到一个他们现在在用的word,

[img]/upload/attachment/44820/1d1e1b2f-2ff2-3080-bd1e-b9212750667b.jpg[/img]
将word里面需要填充的地方使用[替代码]来替代
[img]/upload/attachment/44822/8f78423b-26ad-3c9b-b23c-7a49277d8f59.jpg[/img]
2.在类里面进行hashmap的组装
public static void main(String[] args) throws Exception {

DocGenerator gen = new DocGenerator();
Map data = new HashMap();
data.put("[ORDER_TITLE]", "**科技集团差旅管理服务公司酒店预订单(酒店联)");
data.put("[ORDER_NO]", "HB080709101712996");
data.put("[OPTRID]", "IHZYF011");

//子项
List lstValue = new ArrayList();
Map subData = new HashMap();

subData.put("[CUST_NAME]", "张三");
subData.put("[ROOM_STYLE]", "单床房 大床 宽带 免费");
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]", "双人床房 大床 宽带 免费");
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");//调用处理方法
}
3.处理类书写
/**
* 模板生成类
* @author chm20081009
*
*/
public class DocGenerator {
/**
* 根据传入的数据和模板ID生成
* @author chm
* @param data
* @param templetid
* @throws Exception
*/
@SuppressWarnings("unchecked")
public static String generatorDoc(Map data, String templetid) throws Exception {
String result = "";
// 根据templetid获取模板
String templet = getTemplet(templetid);
//处理模板
templet =split(data,templet);
//去除没有数据的定义变量
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]测试数据";
while (iter.hasNext()) {
String key = iter.next().toString();
Object value = data.get(key);
//替换模板上的标签
templet = replace(key,value,templet);
}
return templet;
}

/**
* 负责替换
* @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) {// 列表
// 从模板中截取迭代定义部分,把整个模板分成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数据
StringBuffer middleResult = new StringBuffer(100);
/**
* 递归调用
*/
for (int i = 0; i < lstValue.size(); i++) {
Map subData = (Map) lstValue.get(i);
String middleData = middleTemplet;
middleData = split(subData,middleData);
middleResult.append(middleData);
}
//将中间替换的数据和拆分开的前后半部分模板组装起来
templet = firstTemplet + middleResult.toString() + lastTemplet;
}

return templet;
}

/**
* 根据模板ID获取模板的内容
* @param templetid
* @return
* @throws Exception
*/
private static String getTemplet(String templetid) throws Exception {
StringBuffer result = new StringBuffer(100);
/*
*可以根据模板ID先到cache里面进行匹配,匹配到模板,直接从缓存中获取
*/

// String fileName = getTempletName(templetid);//写死的文件路径
String fileName = TempletCache.getTemplet(templetid);//从内存根据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.写了一个缓存类
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;
//读取配置文件 模板的路径是存储在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) {
}
}
}

/**
* 根据模板id获取模板路径
* 策略:从内存中根据模板id获得模板路径,如果内存中没有此id则从配置文件中读取该id读出模板路径同时将模板放入内存中
* @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;
}
这里将模板和对应的模板存放路径放入内存中,当需要去生成一个模板的时候,根据已知模板编号(一个类型传真模板是同一个)去从内存中去获取模板存放的路径,如果内存中无该编号,则到配置文件config.properties中根据模板ID去读取对应的模板路径,同时将该路径和模板编号存放在 hashma中

[img]/upload/attachment/46538/a41f3e77-42a1-3222-ad16-a0e6fee4ff3d.jpg[/img]

5.通过DocGenerator.java 里面的递归迭代将模板流中与DATA中key值相同的地方使用该key对应的value替换
这里需要与前面的2做很好的衔接,2做的是组装而这里做的就是解析替换,所有的key必须一致,特别是当模板中有列表的地方

注:当一个word用编辑器打开的时候其代码是很冗长的,我没又没类似Dreamvaver来可视化编辑html一样的工具来编辑word,这是一个头痛的问题,不过如果自己查看,word流也是有他的规律的,什么时候段落是什么时候表格结束什么时候换行,都有他的特定的标识(类似html中<table></table>表示表格,<td></td>表示单元格一样)

6.最终将替换后的流输出保存成文件,就是我们最后的需要的word了

您可能感兴趣的与本文相关的镜像

Stable-Diffusion-3.5

Stable-Diffusion-3.5

图片生成
Stable-Diffusion

Stable Diffusion 3.5 (SD 3.5) 是由 Stability AI 推出的新一代文本到图像生成模型,相比 3.0 版本,它提升了图像质量、运行速度和硬件效率

【数据驱动】【航空航天结构的高效损伤检测技术】一种数据驱动的结构健康监测(SHM)方法,用于进行原位评估结构健康状态,即损伤位置和程度,在其中利用了选定位置的引导式兰姆波响应(Matlab代码实现)内容概要:本文介绍了一种基于数据驱动的结构健康监测(SHM)方法,利用选定位置的引导式兰姆波响应对航空航天等领域的结构进行原位损伤检测,实现对损伤位置与程度的精确评估,相关方法通过Matlab代码实现,具有较强的工程应用价值。文中还提到了该技术在无人机、水下机器人、太阳能系统、四轴飞行器等多个工程领域的交叉应用,展示了其在复杂系统状态监测与故障诊断中的广泛适用性。此外,文档列举了大量基于Matlab/Simulink的科研仿真资源,涵盖信号处理、路径规划、机器学习、电力系统优化等多个方向,构成一个综合性科研技术支持体系。; 适合人群:具备一定Matlab编程基础,从事航空航天、结构工程、智能制造、自动化等相关领域研究的研究生、科研人员及工程技术人员。; 使用场景及目标:①用于航空航天结构、无人机机体等关键部件的实时健康监测与早期损伤识别;②结合兰姆波信号分析与数据驱动模型,提升复杂工程系统的故障诊断精度与可靠性;③为科研项目提供Matlab仿真支持,加速算法验证与系统开发。; 阅读建议:建议读者结合文档提供的Matlab代码实例,深入理解兰姆波信号处理与损伤识别算法的实现流程,同时可参考文中列出的多种技术案例进行横向拓展学习,强化综合科研能力。
【无人机论文复现】空地多无人平台协同路径规划技术研究(Matlab代码实现)内容概要:本文围绕“空地多无人平台协同路径规划技术”的研究展开,重点在于通过Matlab代码实现对该技术的论文复现。文中详细探讨了多无人平台(如无人机与地面车辆)在复杂环境下的协同路径规划问题,涉及三维空间路径规划、动态避障、任务分配与协同控制等关键技术,结合智能优化算法(如改进粒子群算法、遗传算法、RRT等)进行路径求解与优化,旨在提升多平台系统的协作效率与任务执行能力。同时,文档列举了大量相关研究主题,涵盖无人机控制、路径规划、多智能体协同、信号处理、电力系统等多个交叉领域,展示了该方向的技术广度与深度。; 适合人群:具备一定Matlab编程基础和路径规划背景的研究生、科研人员及从事无人机、智能交通、自动化等相关领域的工程技术人员。; 使用场景及目标:①用于学术论文复现,帮助理解空地协同路径规划的核心算法与实现细节;②支撑科研项目开发,提供多平台协同控制与路径优化的技术参考;③作为教学案例,辅助讲授智能优化算法在无人系统中的实际应用。; 阅读建议:建议结合提供的Matlab代码进行实践操作,重点关注算法实现流程与参数设置,同时可参照文中列出的其他相关研究方向拓展技术视野,建议按目录顺序系统学习,并充分利用网盘资源进行仿真验证。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值