昨天看了
@Brin想写程序 的文章
几行Java代码实现的简单模板(不是引擎),呵呵,就非常想去掏掏偶滴小兜兜,果然发现一个类似的东西,因为东西太小,没有准备怎么写,但是看到@Brin想写程序的文章,就想着也发篇文章,说一下当时我的想法与思路。
格式化提供者,用于对字符串进行转换:
1
2
3
4
5
6
7
8
9
10
11
12 |
public interface FormatProvider {
/**
* 把指定的值进行处理后返回
*
* @param string
* 要进行格式化的值
* @return 格式化好的值
* @throws FormatException
*/
String format(Context context, String string) throws FormatException;
}
|
接口方法只有一个,输入有两个参数,一个是上下文,一个是要进行格式的串,返回的值是格式化处理好的串。
当然,我也担心,一些串可能会与我们的点位符有冲突,因此期望能由用户自行指定点位符,因此设定了下面的接口:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65 |
/**
* 模式匹配处理接口
*
* @author luoguo
*
*/
public interface PatternDefine {
/**
* 返回正则匹配
*
* @return
*/
Pattern getPattern();
/**
* 设置前缀
*
* @param prefixPatternString
*/
void setPrefixPatternString(String prefixPatternString);
/**
* 设置后缀
*
* @param postfixPatternString
*/
void setPostfixPatternString(String postfixPatternString);
/**
* 设置正则表达式中间部分
*
* @param patternString
*/
void setPatternString(String patternString);
/**
* 返回正文部分
*
* @param string
* @return
*/
String getPureMatchText(String string);
/**
* 根据正文返回完整部分
*
* @param string
* @return
*/
String getFullMatchText(String string);
/**
* 设置域分隔符
*
* @return
*/
void setSplitChar(char splitChar);
/**
* 返回分隔符
*
* @return
*/
char getSplitChar();
}
|
当然上面的接口如果是固定一个的话,框架内部已经提供,不必另行进行扩展。
格式化接口如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30 |
/**
* 格式化的接口
*
* @author luoguo
*
*/
public interface Formater extends FormatProvider {
/**
* 设置正则表达式,如果不想用默认正则表达式,可以通过此方法自行定义
*
* @param patternHandle
*/
void setPatternHandle(PatternDefine patternHandle);
/**
* 设置格式化提供者
*
* @param formatProviders
* Key为匹配范围符
*/
void setFormatProviders(Map<String, FormatProvider> formatProviders);
/**
* 添加格式化提供者
* @param prefix 前缀
* @param formatProvider
*/
void addFormatProvider(String prefix, FormatProvider formatProvider);
}
|
三个方法, setPatternHandle用于设定格式话模式,setFormatProviders用于设定格式化提供者,由于是一个map,key值是前缀,value是对应的格式化处理器。当然也可以通过addFormatProvider一个一个的增加出来。
好的,接口的事情就搞定了,我们来看看具体的实现类:
默认的格式化实现类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51 |
public class DefaultPatternDefine implements PatternDefine {
private static final String DEFAULT_PATTERN_STRING = "([$]+[{]+[a-zA-Z0-9[.[_[:[/[#]]]]]]+[}])";
private static final String DEFAULT_POSTFIX_PATTERN_STRING = "}";
private static final String DEFAULT_PREFIX_PATTERN_STRING = "${";
private static final char DEFAULT_SPLIT_CHAR = ':';
private String prefixPatternString = DEFAULT_PREFIX_PATTERN_STRING;// 前缀
private String postfixPatternString = DEFAULT_POSTFIX_PATTERN_STRING;// 后缀
private String patternString = DEFAULT_PATTERN_STRING;// 中间部分
private Pattern pattern;
private char splitChar = DEFAULT_SPLIT_CHAR;// 域分隔符
public Pattern getPattern() {
if (pattern == null) {
pattern = Pattern.compile(patternString);
}
return pattern;
}
public void setPrefixPatternString(String prefixPatternString) {
this.prefixPatternString = prefixPatternString;
}
public void setPostfixPatternString(String postfixPatternString) {
this.postfixPatternString = postfixPatternString;
}
public void setPatternString(String patternString) {
this.patternString = patternString;
}
public String getPureMatchText(String string) {
int startPos = prefixPatternString.length();
int endPos = string.length() - postfixPatternString.length();
return string.substring(startPos, endPos);
}
public String getFullMatchText(String string) {
return String.format("%s%s%s", prefixPatternString, string,
postfixPatternString);
}
public void setSplitChar(char splitChar) {
this.splitChar = splitChar;
}
public char getSplitChar() {
return splitChar;
}
}
|
下面是一个针对Context的格式串:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23 |
public class ContextFormater implements FormatProvider {
public String format(Context context, String string) throws FormatException {
Object obj = context.get(string);
if (obj != null) {
return obj.toString();
}
int index = string.indexOf('.');
if (index > 0) {
String name = string.substring(0, index);
obj = context.get(name);
if (obj != null) {
String property = string.substring(index + 1);
try {
return BeanUtils.getProperty(obj, property).toString();
} catch (Exception e) {
throw new FormatException(e);
}
}
}
return null;
}
}
|
下面是核心的格式化算法了:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77 |
public class FormaterImpl implements Formater {
private Map<String, FormatProvider> formatProviders;
private PatternDefine patternDefine = new DefaultPatternDefine();
/**
* 构造函数 使用默认的配置加载器
*/
public FormaterImpl() {
}
/**
* 格式化找到的内容,其余内容不变,如果找不到内容,则原样保留
*
* @throws FormatException
*/
public String format(Context context, String source) throws FormatException {
Matcher matcher = patternDefine.getPattern().matcher(source);
StringBuffer buf = new StringBuffer();
int curpos = 0;
while (matcher.find()) {
String replaceStr = patternDefine.getPureMatchText(matcher.group());
buf.append(source.substring(curpos, matcher.start()));
curpos = matcher.end();
String str = formatSingle(context, replaceStr);
if (str != null) {
buf.append(str);
}
continue;
}
buf.append(source.substring(curpos));
return buf.toString();
}
/**
* 格式化字符串
*
* @param string
* String
* @return String
* @throws FormatException
* @throws Exception
*/
private String formatSingle(Context context, String string)
throws FormatException {
String s[] = string.split(patternDefine.getSplitChar() + "");
if (s.length >= 2) {
FormatProvider o = (FormatProvider) formatProviders.get(s[0]);
if (o != null) {
return o.format(context, s[1]);
}
} else {
FormatProvider o = (FormatProvider) formatProviders.get("");
if (o != null) {
return o.format(context, string);
}
}
return patternDefine.getFullMatchText(string);
}
public void setFormatProviders(Map<String, FormatProvider> formatProviders) {
this.formatProviders = formatProviders;
}
public void setPatternHandle(PatternDefine patternHandle) {
this.patternDefine = patternHandle;
}
public void addFormatProvider(String prefix, FormatProvider formatProvider) {
if (formatProviders == null) {
formatProviders = new HashMap<String, FormatProvider>();
}
formatProviders.put(prefix, formatProvider);
}
}
|
好吧,还有一些配置相关的类,由于不是关键性的,就不在这里讲了,那么接下来看示例:
增加一个常量格式化提供者:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 |
public class ConstFormatProvider implements FormatProvider {
Map<String, String> constMap = new HashMap<String, String>();
public String format(Context context, String key) {
return constMap.get(key);
}
public Map<String, String> getConstMap() {
return constMap;
}
public void setConstMap(Map<String, String> constMap) {
this.constMap = constMap;
}
}
|
再增加一个日期格式化提供者:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 |
public class DateFormatProvider implements FormatProvider {
Map<String, String> constMap = new HashMap<String, String>();
public String format(Context context, String key) {
return constMap.get(key);
}
public Map<String, String> getConstMap() {
return constMap;
}
public void setConstMap(Map<String, String> constMap) {
this.constMap = constMap;
}
}
|
再增加一个用于测试的POJO类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29 |
public class User {
String name;
int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public User() {
}
public User(String name, int age) {
this.name = name;
this.age = age;
}
}
|
好吧,我承认,前面都是做铺垫,跑龙套的,真正的秀场下面开始:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44 |
/**
* 测试不存在任何标记情况
*
* @throws FormatException
*/
public void testFormatNotPlaceholder() throws FormatException {
assertEquals("this is test", formater.format(context, "this is test"));
}
/**
* 测试存在标记,且有处理提供者处理的情况
*
* @throws FormatException
*/
public void testFormatExistPlaceholderProvider() throws FormatException {
Context context = new ContextImpl();
assertEquals("this is v1 test",
formater.format(context, "this is ${const:1} test"));
}
/**
* 测试存在标记,且没有处理提供者处理的情况
*
* @throws FormatException
*/
public void testFormatExistPlaceholderNoProvider() throws FormatException {
assertEquals("this is ${abc:2} test",
formater.format(context, "this is ${abc:2} test"));
}
/**
* 测试存在标记,且是bean的情况
*
* @throws FormatException
*/
public void testFormatBean() throws FormatException {
User user = new User("aa", 123);
context.put("user", user);
assertEquals("this is aa test 123",
formater.format(context, "this is ${context:user.name} test ${context:user.age}"));
}
|
下面总结一下:
上面的格式化占位符方式是${...}方式的,中间的...可以是aa:bb的方式,或者直接是bb的方式,冒号前面实际是一个区域的概念,表示由对应的区域处理器进行处理。这样就可以由开发人员不断的扩展格式化处理器的处理能力。由于占位匹配器也是可以进行扩展的,因此,可以自行定义自己的格式化占位方式。
对于对象的属性可以无限向下“.”下去,当然也可以添加其它的处理方式,比如:数组之类的。
所以从功能及定位来说,与
@Brin想写程序 是一样的。
剧透一下:当时我本来是想写模板语言的,后来直接选择复用Velocity了,所以,就只到此为止了。
虽然放弃了,但是其中在设计及基础构架方面的一些思想及模式,还是值得同学们参考与借鉴的。
分享到:
相关推荐
- **Tiny Formater**:提供了一种格式化数据的方法,有助于开发者在调试过程中更好地理解数据结构。 - **Service Mock服务模拟**:介绍Service Mock的用途,即在没有实际服务可用的情况下模拟服务行为,从而便于进行...
tiny-binary-format, 内存高效二进制格式代替对象 tiny-binary-format使用二进制格式代替对象的内存高效 JS 。var BinaryFormat = require('tiny-binary-format');var Tile = new Bina
TinyXML和TinyXML2是两个小型的开源库,专门用于解析和操作XML(eXtensible Markup Language)文档。这两个库都是用C++编写的,为C++开发者提供了方便的接口来读取、写入和操作XML数据。下面将详细介绍这两个库的...
TinyXML-2,通常以tinyxml2库的形式存在,是一个轻量级的C++库,专门用于解析和操作XML文档。这个库由李·斯坦普(Lee Stemp)开发,设计简洁,易于集成到各种项目中,特别是对于嵌入式系统和资源有限的环境。在给定...
RTX51 TINY 2.02 中文手册知识点总结 本文档对 KEIL 公司的 RTX51 TINY 2.02 版实时操作系统内核进行了详细的介绍和说明。该手册主要面向开发者和工程师,旨在帮助他们快速了解和掌握 RTX51 TINY 2.02 的功能和使用...
TinyXML2是一个小型、快速且易于使用的XML解析库,主要用于C++编程。版本9.0.0是该库的一个更新版本,提供了最新的功能和错误修复。在这个版本中,开发者可以期待更加稳定和优化的XML处理性能。 TinyXML2的设计目标...
《深入理解TINY编译器源码:C与C++编程实践》 TINY编译器,作为一个小型的、易于理解的编译器项目,是学习编译原理和实践的重要资源。它用C和C++编写,源码简洁且带有详细注释,非常适合初学者或者对编译器感兴趣的...
TinyXML2是一个轻量级的XML解析库,主要用于读取和写入XML文档。这个库在C++中设计,提供了一种简洁的方式来处理XML数据,适用于小型项目或嵌入式系统。在“TinyXML2使用示例”中,我们可以通过提供的源代码和XML...
### TinyOS内核分析 #### 1. 综述 TinyOS是一款专为无线传感器网络设计的操作系统,其内核采用非抢占式设计,这意味着它依赖于任务的主动放弃CPU控制权来完成任务间的切换。这种设计使得TinyOS能够更好地支持资源...
tiny 编译器关于 while 和 for finish 语句编译的扩充 本资源主要介绍了tiny编译器关于while和for finish语句编译的扩充,包括词法分析、语法分析、语义分析、代码生成等部分。tiny编译器是一个简化的编译器,省略...
YOLOv7-Tiny是YOLO(You Only Look Once)系列目标检测算法的一个轻量级变体,由Alexey Bochkovskiy等开发者在Darknet框架下设计。这个算法在保持高效的同时,具备较高的目标检测精度。在给定的压缩包中,包含了两个...
TinyXML2是一个轻量级的XML解析库,主要用于读取和写入XML文档。这个库由Steve Baker开发,它的最新版本在2017年8月17日更新,这表明了开发者持续对库进行维护和改进以适应不断变化的编程需求。在本文中,我们将深入...
QT框架是广受欢迎的开源C++开发平台,用于构建跨平台的应用程序,而TinyXML则是一个小巧、轻量级的开源库,专门用于解析和操作XML文档。在QT项目中集成TinyXML,可以方便地处理XML数据,尤其适用于那些需要读取、...
TinyXML、TinyXPath和TinyXML++是开源的C++库,专门用于解析和操作XML文档。这些库在处理XML数据时提供了轻量级且高效的选择,尤其适合嵌入式系统和资源有限的环境。 TinyXML,版本2.5.3,是一个小型、简单且易于...
**tiny-dnn源码分析** tiny-dnn是一个轻量级的、C++11实现的深度学习框架,它旨在提供高效、易于理解和部署的神经网络解决方案。作为一个开源项目,tiny-dnn为开发者提供了深入理解深度学习算法和实现的绝佳机会。...
根据提供的信息,本部分将围绕Tiny4412开发板的使用、更新及应用场景进行深入探讨,涵盖嵌入式Linux、C语言、ARM架构等多个技术点。 一、Tiny4412开发板概述 Tiny4412是一款基于ARM9架构的开发板,它为嵌入式Linux...
TinyXML2是一个轻量级的XML解析库,主要由C++编写,旨在提供简单、高效的方式来读取和写入XML文档。它适用于各种平台,包括桌面系统和移动系统,如Android。这个库的设计目标是小巧且易于理解,使得开发者能够快速地...
### TinyXML入门教程知识点详解 #### 一、XML简介与TinyXML概述 - **XML**:全称为**可扩展标记语言**(EXtensible Markup Language),是一种用于标记数据的语言,允许用户自定义数据的标识,从而区分不同类型的结构...
Tinycore v6.3 是一个基于Linux的极小型操作系统,其设计目标是提供一个轻量级、快速启动的环境,适合于老旧硬件、虚拟化应用、服务器或作为基础系统进行定制开发。Tinycore Linux 以其小巧的体积和高度模块化的特性...
《深入解析Darknet YOLOv7-tiny预训练模型》 在计算机视觉领域,目标检测技术占据着举足轻重的地位,而YOLO(You Only Look Once)算法以其高效和精准的特点备受青睐。YOLOv7-tiny是YOLO系列的最新版本之一,它在...