FreeMarker是一款基于java的模版引擎,用于生成文本(比如HTML)。它的语法和作用和JSP类似,JSP需要依赖Servlet容器(如Tomcat),而FreeMarker只需要Java运行环境。
官方网站上的这张图很形象的说明了FreeMarker的用途和工作原理:模版+数据=文本(HTML)
在Jspxcms中,模版就是/template/1/default/
目录下的模版文件(后缀也是.html
),数据来自SpringMVC Controller的Model.addAttribute(String name, Object value)
或者Jspxcms自定义标签(如:[@InfoList node='news';list]...[/@InfoList]
)。
更多资料可以参考FreeMarker官方网站的文档,FreeMarker官方还提供了FreeMarker中文文档。
和JSP的EL表达式基本是一样的:${myname}
,${user.username}
。
${customs['abc']}
${arr[0]}
+ - * / %
。如:${100 – x*x} ${x/2} ${12%10}
< <= >= > == = !=
或者 lt lte gt gte
|| && !
内建函数是FreeMarker的一大优势,很多在JSP里面需要通过复杂处理才能实现的功能,这里只需要用简单的内建函数就搞定了。
${mouse!"No mouse"}
${username!"匿名用户"}
。FreeMarker中输出null值会报错,如果希望对象为null时,什么都不显示且不报错,可以这样处理 ${mouse!}
${user.username!}
${(user.username)!}
(最后一种方式可以避免user对象为null导致的错误)。${foo?string("yes", "no")}
${lastUpdated?string("yyyy-MM-dd HH:mm:ss")}
${username?html}
。为避免直接输出<
>
等值,导致XSS攻击,通常会对输出的值进行转义。${foo?js_string}
。对js中的引号等字符进行处理,给js变量赋值是非常有用,比如var s = "${foo?js_string}"
。${'abc'?substring(2)}
${username?substring(0,3)}
${'abc'?length}
${username?length}
${list?size}
${"GrEeN MoUsE"?lower_case}
${"GrEeN MoUsE"?upper_case}
FreeMarker标签类似JSP标签。标签默认使用尖括号< >
,在Jspxcms中为了避免和HTML标签混淆,便于在Dreamweaver中编辑,使用中括号[ ]
作为标签符号。以下示例一律使用中括号。
标签有两种,一种是系统自带标签,以[#
开头;一种是自定义标签,以[@
开头。
注释标签:[#-- 这是需要注释的代码 --]
以InfoList标签为例说明。
[@InfoList node='news';infos]
[#list infos as info]
<a href="${info.url}">${info.title}</a>
[/#list]
[/@InfoList]
[@InfoList node='news';list]
[#list list as bean]
<a href="${bean.url}">${bean.title}</a>
[/#list]
[/@InfoList]
[@InfoList;list]
[#list list as bean]
<a href="${bean.url}">${bean.title}</a>
[/#list]
[/@InfoList]
InfoList
。用于获取文档列表的标签。node='news'
。node
是参数名,'news'
是参数值。意为获取栏目代码为news
的文档。有时候参数不是必须的。;infos
。分号;
后面的是返回值。infos
是标签返回的对象。标签获取的文档列表就存放在这个对象里,这个对象的名称可以随意定义。[#if 2>1]
...
[#elseif username=="abc"]
...
[#elseif username?starts_with("red")]
...
[#else]
...
[/#if]
判断是否为null:[#if username??]...[/#if]
[#list sequence as item]
...
[/#list]
[#list 1..10 as i]
...
[/#list]
${item_index}
${item_index + 1}
。[#if item_has_next]...[/#if]
sequence as item
和item_index
、item_has_next
中的item
必须一致,如果sequence as info
,则用${info_index}
、${info_has_next}
。[#list seq as x]
${x}
[#if x = "spring"][#break][/#if]
[/#list]
[#include "/common/copyright.ftl"]
[#include "/common/navbar.html" parse=false /]
[#assign myname="abc"]
[#assign myname=username]
[#assign myname]孔子[/#assign]
为了避免跨站脚本攻击(XSS),通常会对输出的内容做HTML转义,比如${foo?html}。但是所有变量都要做这个转义不仅麻烦,还容易遗忘。另外FreeMarker空值处理也很麻烦且容易遗忘,比如${foo!}
、${(user.username)!}
。
使用excape标签可以很好的解决这个问题。
[#escape x as (x)!?html]
...
${user.username}
...
[/#escape]
只要被这个标签包含的代码,都相当于加上了${(foo.bar)!?html}
,如${user.username}
相当于${(user.username)!?html}
。即包含了空值处理,也包含了HTML转义处理。
在escape标签内有对象不需要转义时,可以用noescape标签。
[#escape x as (x)!?html]
...
[#noescape]${text}[/#noescape]
...
[/#escape]