1. 模板+数据模型=输出
2. 数据模型
a) 类似目录的变量称为hashes,包含保存下级变量的唯一的查询名字,充当其它对象的容器,每个都关联唯一的查询名字
b) 类似文件的变量称为scalars,保存单值,有两种类型
i.
字符串:用引号括起,单引号or双引号
ii. 数字:不用引号
iii. 日期:可以是日期,时间或日期-时间
iv. 布尔值:true或false,通常在<#if …>标记中使用
v. 对scalars的访问从root开始,各部分用”.”分隔,如 animals.mouse.price
c) 另外一种变量是sequences,和hashes类似,充当其它对象的容器,按次序访问,只是不使用变量名字,而使用数字索引,如animals[0].name,索引值从0开始
d) 通常每个变量具有上述的一种能力,但一个变量可以具有多种上述能力
e) 方法
i. ${avg(3,5,20)} ${avg(student.zhangyaang.age,student.situ.age)}
f) 宏和变换器:用户
自定义的指令(FTL标记)
g) 节点:树形结构中的一个节点,常用于xml处理中
3. 模板
a) 文本:直接输出
b) ${…}:称为interpolations,FreeMarker会在输出时用实际值代替,或#{},只用于文本中
c) FTL标记:类似于HTML,为了与HTML区分,用#开始,有的以@开始,不会输出,区分
大小写,FTL标记不能位于另一个FTL标记内部
d) 注释:<#--和-->,不会输出
e) 多余的空白字符会在模板输出时移除
f) 指令
i. 使用FTL标记引用指令,有三种FTL标记
1. 开始标记:<#name prarm>
2. 结束标记:</#name>
3. 空内容指令标记:<#name param />
ii. 有两种类型的指令:预定义指令和用户定义指令,用户定义指令要用@替换#
iii. FTL标记不能够交叉,应该正确嵌套
iv. 如果使用不存在的指令,FreeMarker会产生错误消息
v. < , </ 和指令间不允许有空白字符
g)
表达式
i. 字符串
1. 使用单引号或双引号限定
2. 特殊字符要转义
3. raw字符串,可以认为是文本,其中的$和{等不具有特殊含义,该类字符串在引号前面加r,如${r”${foo}”} ${r”c:foobar”}
ii. 数字
1. 直接输入,不需要引号
2. 精度数字使用”.”分隔,不能使用分组符号
3. 不支持科学计数法
4. 不能省略小数点前面的0
5. 数字8 ,+8 ,08和8.00 都是相同的
iii. 布尔值:true 和 false, 不使用引号
iv. 序列
1. 有逗号分隔的子变量列表,有方括号限定,如:
<#list [“winter”, ”spring”, ”summer”, ”autumn”] as x>
${x}
</#list>
2. 列表的项目是表达式,如:
[2+2, [1,2,3,4], ”whatnot”]
3. 可以使用数字范围定义数字序列,如:
2..5等同于[2,3,4,5] 注意:数字范围没有方括号 如:5..2
4. 散列(hash) 由逗号的键/值列表,有大括号限定,键和值之间用冒号分隔
{“name”:”zhangyang”,”age”:23} 键和值都是表达式,但键必须是字符串
v. 获取变量
1. 顶层变量:${variable}, 变量名只能是字母, 数字,
下划线, $, @ 和# 的组合, 不能以数字开头
2. 从散列中获取数据
a) school.student.name
b) school.[“student”].name
c) school[“student”][“name”]
说明:使用点语法,变量名有顶层变量一样的限制, 方括号语法没有该限制
3. 从序列获(sequences)得数据: 和散列的方括号语法一样, 只是方括号的表达式值必须是数字; 注意: 第一个项目的索引是0
4. 序列片段: 使用[startIndex..endIndex]语法, 从序列中获得片段(也是序列)
5. 特殊变量: FreeMarker内定义的变量, 使用 .variablenae 语法访问
vi. 字符串操作
1. Interpolation(或连接操作)
a) 可以使用${..} 或#{..} 在文本部分插入表达式的值,
如${“Hello ${user}”} 等于 ${“Hello”+user}
b) ${..} 只能用于文本部分, 不用于FTL表达式中
2. 子串
假设user的值为”tom cat”
${user[0]}${user[4]} ? tc
${user[1..4]} ? om c
vii. 序列操作
连接操作: 和字符串一样, 用”+”
<#list [“joe”,”fred”] + [“julia”,”kate”] as user>
${user}
</#list>
viii. 散列操作
连接操作: 和字符串一样, 用”+”, 如果有相同的key, 右边的值替换左边的值(后面的替换前面的), 如
<#assign ages = {“joe”:23,”fred”:25} + {“joe”:30,”julia”:23}>
</#assign> 结果: joe的值为30
ix. 算术运算
1. +, -- , *, /, %
${x / 2}
2.
操作符两边必须是数字
3. 使用 “+” 操作符时,如果一边是数字, 一边是字符串, 就会自动将数字转换为字符串, 如: ${3+”4”} 结果 34
4. 使用内建的 int 获得整数部分
${1.1?int} ?1
${1.999?int} ?1
${(5/2)?int} ?2
5. 比较操作符
a) 使用=(或==,完全相等)测试两个值是否相等, 使用!=测试两个值是否不相等
b) = 和!= 两边必须是相同类型的值
c) FreeMarker是精确比较,”x”,”x “和”x”是不等的
d) 对数字和日期可以使用<, <=, >, 和>=, 但不用用于字符串
e) 由于FreeMarker 会将>解释成FTL标记的结束字符, 所以对于>和>=可以使用括号来避免这种情况, 如<#if (x > y)>
f) e)的一种替代方法, 使用lt, lte, gt 和gte来替代
6. 逻辑操作符
a) &&(and), ||(or), !(not), 只能用于布尔值
<#if x < 13 && color = “green”>
…
</#if>
<#if ! man>
woman
</#if>
7. 内建函数
a) 内建函数的用法类似访问散列的子变量, 只是用”?”|替代”,”
b) 用于字符串
i. html: 对字符串进行html编码
ii. cap_first: 将字符串转换成第一个字母大写
iii. lower_case: 小写
iv. upper_case: 大写
v. trim: 去掉字符串前后的空白字符
c) 用于序列
i. size: 获得序列中元素的数目
d) 用于数字
i. int: 取得数字的整数部分
e) 例子:
test = “tom & jerry”
${test?html} ? tom & jerry
${test?upper_case?html} ? TOM & JERRY
8. 操作符优先顺序
h) Interpolation(只用于文本部分)
i. Interpolation有两种类型
1. 通用Interpolation: ${expr}
a) 插入字符串值: 直接输出表达式结果
b) 插入数字值: 根据缺省格式(由#setting指令设置)将表达式转换成文本输出; 可以使用内建函数string格式化单个Interpolation
c) 例子
<#setting number_format=”currency”/> ?金额
<#assign answer=43/>
${answer} ? $43.00
${answer?string} ?$43.00
${answer?string.number} ?43
${answer?string.currency} ?$43.00
${answer?string.percent} ?4,300%
2. 数字Interpolation: ${expr} 或#{expr ; format}
3. 插入日期: 根据缺省格式(#setting) 将表达式结果转换成文本输出; 可以使用内建的函数string 格式化单个Interpolation, 例子:
${update?string(“yyyy-MM-dd HH:mm:ss zzzz”)}
4. 插入布尔值: 说明同上
<#assign foo = true/>
${foo?string(“yes”,”no”)} ?yes
5. 数字Interpolation的#{expr ; format}形式可以用来格式化数字, format可以是
a) mX: 小数部分最小X位
b) MX: 小数部分最大X位
c) 例子
<#assign x=2.364/>
<#assign y=4/>
#{x; M2} ?2.36
#{y; M2} ?4
#{x; m1} ?2.4
#{y; m1} ?4.0
i) 例子
i. if指令
<#if zhangyang.age < situ.age>
Situ is order
<#else>
Zhangyang is order
</#if>
ii. list指令 <#list students[0..5] as s_list></#list>取list的前5条记录
<table border = “1”>
<tr>
<td>name</td><td>age</td>
<#list students as s_list>
<td>${s_list.name}</td><td>${s_list.age}</td>
</#list>
</tr>
</table>
iii. include指令
<#include “/all/header.html”>
…
<#include “/all/footer.html”>
iv. 应用
<table>
<#list students as s_list>
<tr>
<td>
<#if s_list.sex =”man”><img src=”/img/boy.gif”/>
<#else><img src=”/img/girl.gif/>
</#if>
${s_list.name}
</td>
<td>${s_list.age}</td>
</tr>
</#list>
</table>