常用的模板引擎:
- jinja
- freemarker
- velocity
基本能力是差不多的。只不过jinja是python实现,其他两个是java实现。
freemarker用法
<#--普通变量替换 -->
package ${classPath};
public class ${className} {
public static void main(String[] args) {
System.out.println("${helloWorld}");
}
}
<#--函数 -->
第一个字母大写:${data?cap_first}
所有字母小写:${data?lower_case}
所有字母大写:${data?upper_case}
数值取整数:${floatData?int}
<#--列表操作 -->
获取集合的长度:${users?size}
<#if users??>
<#list users as user >
${user}
</#list>
<#else>
${user!"变量为空则给一个默认值"}
</#if>
<#--字典操作 -->
直接通过Key获取 Value值:${mapData.liping}
直接通过Key获取 Value值:${mapData['liping']}
通过Key遍历Map:
<#list mapData?keys as key>
${key} -> ${mapData[key]}
</#list>
通过Value遍历Map:
<#list mapData?values as value>
${value}
</#list>
注意几点:
- 待替换的变量必须写成“${变量名}”形式,大括号不能省略
- 函数调用使用?,而且支持哪些函数都是freemarker内置的
处理上述模板的java代码如下:
static void testFreeMarker() throws IOException, TemplateException {
Configuration configuration = new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS);
Template template = configuration.getTemplate("src/main/resources/hello.ftl");
Map<String, Object> dataMap = new HashMap<String, Object>();
dataMap.put("classPath", "com.freemark.hello");
dataMap.put("className", "AutoCodeDemo");
dataMap.put("helloWorld", "HelloWorld!");
dataMap.put("data", "abcd1234");
dataMap.put("floatData", 12.34);
List<String> users = Arrays.asList("bob", "suzy");
dataMap.put("users", users);
Map<String, Integer> m = new LinkedHashMap<>();
m.put("liping", 30);
m.put("xx1", 7);
m.put("xx2", 56);
dataMap.put("mapData", m);
try(Writer writer = new FileWriter("hello.res"))
{
template.process(dataMap, writer);
}
}
Velocity用法
velocity语法参考:
https://www.ibm.com/developerworks/cn/java/j-lo-velocity1/
我们来写出与上述模板等价的velocity模板:
## 普通变量替换,与freemarker不同,大括号不是必须的
package ${classPath};
public class $className {
public static void main(String[] args) {
System.out.println("${helloWorld}");
}
}
## 函数
第一个字母大写:${Helper.capitalize($data)}
所有字母小写:${data.toLowerCase()}
所有字母大写:${data.toUpperCase()}
数值取整数:${Helper.toInt($floatData)}
## 列表操作
获取集合的长度:${users.size()}
#if($users)
#foreach($user in $users)
${user}
#end
#else
${users}
#end
## 字典操作
直接通过Key获取 Value值:${mapData.liping}
直接通过Key获取 Value值:${mapData['liping']}
通过Key遍历Map:
#foreach($item in ${mapData.entrySet()})
${item.key} -> ${item.value}
#end
通过Value遍历Map:
#foreach($value in $mapData)
${value}
#end
与freemarker的差异:
- 引用变量时大括号不是必须的,像$className在freemarker里就是无效的
- 函数调用使用点号,且支持哪些函数使用的是java的反射机制,像字符串大写使用${data.toLowerCase()}
- 列表、字典遍历使用#foreach
- 可以传递一个类给velocity,在模板里调用这个类的静态方法,像“第一个字母大写”用到的{Helper.capitalize( {Helper.capitalize( data)},Helper是我们自己提供的一个类,capitalize是其静态方法。
处理velocity模板的java代码如下:
static void testVelocity() throws IOException {
VelocityEngine ve = new VelocityEngine();
ve.setProperty(RuntimeConstants.RESOURCE_LOADER, "classpath");
ve.setProperty("classpath.resource.loader.class", ClasspathResourceLoader.class.getName());
ve.init();
//这里要指定utf8编码,否则中文输出乱码
org.apache.velocity.Template t = ve.getTemplate("hello.vtl", "utf-8");
VelocityContext dataMap = new VelocityContext();
dataMap.put("classPath", "com.freemark.hello");
dataMap.put("className", "AutoCodeDemo");
dataMap.put("helloWorld", "HelloWorld!");
dataMap.put("data", "abcd1234");
dataMap.put("floatData", 12.34);
dataMap.put("Helper", Helper.class);
List<String> users = Arrays.asList("bob", "suzy");
dataMap.put("users", users);
Map<String, Integer> m = new LinkedHashMap<>();
m.put("liping", 30);
m.put("xx1", 7);
m.put("xx2", 56);
dataMap.put("mapData", m);
try(Writer writer = new FileWriter("hello_v.res"))
{
t.merge(dataMap, writer);
}
}
我们还要提供Helper辅助类,才能确保模板输出正确:
public final class Helper
{
public static String capitalize(String s)
{
if (s == null || s.length() == 0)
{
return s;
}
char[] charArray = s.toCharArray();
charArray[0] -= 32;
return String.valueOf(charArray);
}
public static int toInt(double d)
{
return (int)d;
}
}
注意:Helper类必须要声明为public,否则模板里无法访问到。
jinja用法
jinja语法可参考:
http://docs.jinkan.org/docs/jinja2/templates.html#builtin-globals
上述模板等价的jinja模板为:
{# 普通变量替换 #}
package {{classPath}};
public class {{className}} {
public static void main(String[] args) {
System.out.println("{{helloWorld}}");
}
}
{# 函数 #}
第一个字母大写:{{data | capitalize}}
所有字母小写:{{data | lower}}
所有字母大写:{{data | upper}}
数值取整数:{{floatData | int}}
{# 列表操作 #}
获取集合的长度:{{users | length}}
{% if users %}
{% for user in users %}
{{user}}
{% endfor %}
{% else %}
{{users}}
{% endif %}
{# 字典操作 #}
直接通过Key获取 Value值:{{mapData.liping}}
直接通过Key获取 Value值:{{mapData['liping']}}
通过Key遍历Map:
{% for key in mapData.keys() %}
{{key}} -> {{mapData[key]}}
{% endfor %}
通过Value遍历Map:
{% for value in mapData.values() %}
{{value}}
{% endfor %}
与velocity、freemarker的差别:
- 使用{{}}来引用变量
- 使用过滤器来模拟函数调用
处理模板的python代码:
from jinja2 import Environment, FileSystemLoader
env = Environment(loader=FileSystemLoader('./', encoding='utf-8'))
template = env.get_template('hello.jtl')
cont = template.render(classPath='com.freemark.hello', className='AutoCodeDemo',
helloWorld=u'HelloWorld!',
data='abcd1234',
floatData=12.34,
users=["bob", "suzy"],
mapData={"liping" :30, "xx1":7, "xx2" : 56})
# print cont
with open('hello_j.res', 'w') as writer:
writer.write(cont.encode('utf-8'))
总结
freemarker、jinja更像DSL,比较简洁,而velocity则像在写java代码,有点啰嗦。