还是首先讲一下需求。一个普通的web form表单提交,根据模板自动生成指定格式的结果。form的优势在格式化数据,使得各属性非常直观的展现出来,用户可以更加简单直观的进行输入。但业务上的最终结果却不可以是form,所以就有了这个需求。需求的本质有点类似el表达式的替换,但是这个表达式模板是动态配置的,而不是常见的XML静态文件。
总结一下需求,概括来讲是这样:根据用户的输入,将业务属性填充到实时设置的模板生成最终结果。
不难发现这里的几个关键点。
模板要实时可以配置,这里采用db方式。存在用户输入的行为,也就意味着存在不稳定因素,包括特殊字符,空等。但是因为是填充,可以过滤掉特殊字符,只要处理null即可。(需求上需要处理null)既然是填充,就要保证两点。一是填充的对象不能混淆错乱,二是填充的顺序不能出错。该模板是当做第三方jar依赖注入的,所以必须规避掉任何业务因素。 想清楚了设计的重点,再来看看设计,先看类图。
实体层: TemplateEl:el表达式的设计。因为只是简单的文本模板,所以只要关心el的前缀、后缀即可。这里的html配置是为了前台效果展示、编辑用的。propertyMethod是考虑到不同系统、程序员在声明getter、setter方法时可能不一致,所以显示表达了。 TemplateElFormat:针对特殊el属性,有一定的格式约定。我目前只用到了时间格式,后面会有介绍。 Template:常见的group+unique code 唯一标示模板。 TemplateElConfig:就是一张普通的mapping表。 ResultVo:根据业务需求返回指定的VO。这里建议用一个Abstract类来为模板服务。
Service层:
TemplateFactory:常规的工厂类,获取指定生成器。
TemplateGenerator:常规的生成器。
还是老话,结构jar提供,实现在业务层。包括vo。这样的好处是jar与业务完全隔离。坏处是每个业务系统都要写一遍实现,而且存在冲突的风险。
下面补上实现类的generator实现,其他代码没什么特别。
1. Exception是模板jar封装过的几类异常。因为不存在业务代码,所以无法控制调用方的传参,模板可能会不存在。
//1. 获取模板
Template template = this.templateLogic.findByGroupAndCode(groupCode, templateCode);
if (null == template) {
logger.info("Invalid template Access. group code:{}, template code:{}", groupCode, templateCode);
result.setException(new TemplateNotExistException("Template not exists! Group code:" + groupCode + ",template code:" + templateCode) );
return;
}
2. 读取模板配置的el。如果没有任何配置,warning。这里按seq读取,为模板拼接做准备。
//2. 读取模板配置
List templateElConfigs = this.templateElConfigLogic.findByTemplateId(template.getId());
if (null == templateElConfigs || templateElConfigs.iSEMpty() ) {
logger.info("There's no express configuration for template:{}", template.getName());
return;
}
3. 解析el的配置,生成最终字符串。
for (TemplateElConfig templateElConfig : templateElConfigs) {
TemplateEl el = this.templateElLogic.findOne(templateElConfig.getElId());
if (el == null) {
logger.info("Missing el config, template el config id:{}",templateElConfig.getId());
continue;
}
String datasourceValue = "";
//3.1
String methodName = el.getPropertyMethod();
if (StringUtils.isEmpty(methodName)) {
logger.info("Missing property method config for el:{}",el.getId());
continue;
}
try {
Method method = null;
Object propertyValue;
if (el.getEl().contains("Date")) {
method = (Method)datasource.getClass().getMethod(methodName);
propertyValue = (Date)method.invoke(datasource);
}else {
method = (Method)datasource.getClass().getMethod(methodName);
propertyValue = (String)method.invoke(datasource);
}
if (propertyValue instanceof Date) {
TemplateElFormat elFormat = this.templateElFormatLogic.findByElId(el.getId());
String timeFormat = DEFAULT_DATE_FORMAT;
if (null != elFormat) {
timeFormat = elFormat.getFormat();
}
SimpleDateFormat format = new SimpleDateFormat(timeFormat);
datasourceValue = format.format(propertyValue);
}else if (propertyValue instanceof String) {
datasourceValue = (String)propertyValue;
}
} catch (Exception e) {
logger.error("No such method.", e);
result.setException(new IllegalTemplateConfigException("No such method.", e));
return;
}
if (StringUtils.isNoneEmpty(datasourceValue)) {
if (StringUtils.isNoneEmpty(el.getPrefix())) {
buffer.append(el.getPrefix());
}
buffer.append(datasourceValue);
if (StringUtils.isNotEmpty(el.getSuffix())) {
buffer.append(el.getSuffix());
}
if (StringUtils.isNoneEmpty(el.getHtmlPrefix())) {
htmlBuffer.append(el.getHtmlPrefix());
}
htmlBuffer.append(datasourceValue);
if (StringUtils.isNotEmpty(el.getHtmlSuffix())) {
htmlBuffer.append(el.getHtmlSuffix());
}
}
}
这里没有业务逻辑,所以是可以放到jar里面的,各业务系统只要控制如何结构化调用就行了。
下一篇:关于java数组的返回
¥498.00
¥399.00
¥29.00
¥299.00