zoukankan      html  css  js  c++  java
  • freemarker模版引擎技术总结

    FreeMarker语言概述

    FreeMarker是一个模板引擎,一个基于模板生成文本输出的通用工具,使用纯Java编写。

    FreeMarker被设计用来生成HTML Web页面,特别是基于MVC模式的应用程序

    虽然FreeMarker具有一些编程的能力,但通常由Java程序准备要显示的数据,由FreeMarker生成页面,通过模板显示准备的数据(如下图)

    FreeMarker不是一个Web应用框架,而适合作为Web应用框架一个组件。

    FreeMarker与容器无关,因为它并不知道HTTP或Servlet;FreeMarker同样可以应用于非Web应用程序环境。

    FreeMarker更适合作为Model2框架(如Struts)的视图组件,你也可以在模板中使用JSP标记库。

    FreeMarker是免费的。

    FreeMarker特性

    1、通用目标

    能够生成各种文本:HTML、XML、RTF、Java源代码等等

    易于嵌入到你的产品中:轻量级;不需要Servlet环境

    插件式模板载入器:可以从任何源载入模板,如本地文件、数据库等等

    你可以按你所需生成文本:保存到本地文件;作为Email发送;从Web应用程序发送它返回给Web浏览器

    2、强大的模板语言

    所有常用的指令:include、if/elseif/else、循环结构

    在模板中创建和改变变量

    几乎在任何地方都可以使用复杂表达式来指定值

    命名的宏,可以具有位置参数和嵌套内容

    名字空间有助于建立和维护可重用的宏库,或者将一个大工程分成模块,而不必担心名字冲突

    输出转换块:在嵌套模板片段生成输出时,转换HTML转义、压缩、语法高亮等等;你可以定义自己的转换

    3、通用数据模型

    FreeMarker不是直接反射到Java对象,Java对象通过插件式对象封装,以变量方式在模板中显示

    你可以使用抽象(接口)方式表示对象(JavaBean、XML文档、SQL查询结果集等等),告诉模板开发者使用方法,使其不受技术细节的打扰

    4、为Web准备

    在模板语言中内建处理典型Web相关任务(如HTML转义)的结构

    能够集成到Model2 Web应用框架中作为JSP的替代

    支持JSP标记库

    为MVC模式设计:分离可视化设计和应用程序逻辑;分离页面设计员和程序员

    5、智能的国际化和本地化

    字符集智能化(内部使用UNICODE)

    数字格式本地化敏感

    日期和时间格式本地化敏感

    非US字符集可以用作标识(如变量名)

    多种不同语言的相同模板

    6、强大的XML处理能力

    <#recurse> 和<#visit>指令(2.3版本)用于递归遍历XML树。在模板中清楚和直觉的访问XML对象模型。开源论坛 JForum 就是使用了 FreeMarker 做为页面模板。

     

    入门FreeMarker程序

     1 package demo;
     2 
     3 import freemarker.template.Configuration;
     4 import freemarker.template.Template;
     5 import java.io.File;
     6 import java.io.OutputStreamWriter;
     7 import java.io.Writer;
     8 import java.util.*;
     9 
    10 /**
    11  * @author jiaqing.xu@hand-china.com
    12  * @version 1.0
    13  * @name
    14  * @description
    15  * @date 2017/9/6
    16  */
    17 public class Test1 {
    18     public static void main(String[] args) throws Exception {
    19         //创建Freemarker配置实例
    20         Configuration cfg = new Configuration();
    21         //加载模版的文件目录
    22         cfg.setDirectoryForTemplateLoading(new File("templates"));
    23 
    24         //创建数据模型
    25         Map root = new HashMap();
    26         root.put("user", "老徐");
    27         root.put("date1", new Date());
    28         root.put("num0",90);
    29         
    30         List<Integer> mylist = new ArrayList<>();
    31         mylist.add(1);
    32         mylist.add(2);
    33         mylist.add(3);
    34         root.put("list",mylist);
    35 
    36         root.put("random", new Random().nextInt(100));
    37 
    38         List list = new ArrayList();
    39         list.add("中国");
    40         list.add("美国");
    41         list.add("英国");
    42         root.put("lst", list);
    43         //加载模板文件
    44         Template t1 = cfg.getTemplate("a.ftl");
    45 
    46         //显示生成的数据将合并后的数据打印到控制台
    47         Writer out = new OutputStreamWriter(System.out);
    48         t1.process(root, out);
    49         out.flush();
    50     }
    51 }
     1 <#--测试引入其他模版文件-->
     2 <#import "b.ftl" as bb  />
     3 <@bb.copyright date="2010-2011" />
     4 ${bb.mail}
     5 <#--定义变量并赋值-->
     6 <#assign mail="my@163.com"  />
     7 ${mail}
     8 <#assign mail="my@163.com" in bb  />
     9 ${bb.mail}
    10 
    11 <#--直接获取变量的值-->
    12 你好啊,${user},今天你的精神不错!
    13 今天的日期为:${date1?string("yyyy-MM-dd HH:mm:ss")}
    14 <#--访问list集合-->
    15 ${list[0]}
    16 
    17 <#--if判断-->
    18 ${user}是<#if user=="老徐">我们的老师</#if>
    19 
    20 <#--if else测试-->
    21 <#if num0 gt 18>
    22 及格!
    23 <#else>
    24 不及格!
    25 </#if>
    26 
    27 <#--if elseif else测试-->
    28 <#if random gte 90>
    29 优秀!
    30 <#elseif random gte 80>
    31 良好!
    32 <#else>
    33 一般!
    34 </#if>
    35 
    36 <#--循环遍历List-->
    37 <#list lst as dizhi >
    38 <b>${dizhi[0..1]}</b><br/>
    39 </#list>
    40 
    41 <#assign number="01234">
    42 ${number[0]} <#-- 输出字符0 -->
    43 ${number[0..3]} <#-- 输出子串“0123” -->
    44 
    45 <#--内置函数格式化输出-->
    46 <#assign answer=42/>
    47 ${answer}
    48 ${answer?string} <#-- the same as ${answer} -->
    49 ${answer?string.number}
    50 ${answer?string.currency}
    51 ${answer?string.percent}

    b.ftl文件内容:

    1 <#--定义宏变量-->
    2 <#macro copyright date>
    3 <p>Copyright (C) ${date} 北京尚学堂.</p>
    4 </#macro>
    5 <#--定义变量并赋值-->
    6 <#assign mail = "3072966990@qq.com">

    数据类型

    一、 直接指定值

    直接指定值可以是字符串、数值、布尔值、集合及Map对象。
    1. 字符串
    直接指定字符串值使用单引号或双引号限定。字符串中可以使用转义字符”"。如果字符串内有大量的特殊字符,则可以在引号的前面加上一个字母r,则字符串内的所有字符都将直接输出。

    2. 数值
    数值可以直接输入,不需要引号。FreeMarker不支持科学计数法。

    3. 布尔值
    直接使用truefalse,不使用引号。

    4. 集合
    集合用中括号包括,集合元素之间用逗号分隔。
    使用数字范围也可以表示一个数字集合,如1..5等同于集合[1, 2, 3, 4, 5];同样也可以用5..1来表示[5, 4, 3, 2, 1]

    5. Map对象
    Map对象使用花括号包括,Map中的key-value对之间用冒号分隔,多组key-value对之间用逗号分隔。
    注意:Map对象的keyvalue都是表达式,但key必须是字符串。

     

    6. 时间对象

    root.put("date1", new Date());

    ${date1?string("yyyy-MM-dd HH:mm:ss")}

     

    7. JAVABEAN的处理

    Freemarker中对于javabean的处理跟EL表达式一致,类型可自动转化!非常方便!

    二、 输出变量值

    FreeMarker的表达式输出变量时,这些变量可以是顶层变量,也可以是Map对象的变量,还可以是集合中的变量,并可以使用点(.)语法来访问Java对象的属性。

    1. 顶层变量
    所谓顶层变量就是直接放在数据模型中的值。输出时直接用${variableName}即可。

    2. 输出集合元素
    以根据集合元素的索引来输出集合元素,索引用中括号包括。如: 输出[“1”“2”“3”]这个名为number的集合,可以用${number[0]}来输出第一个数字。FreeMarker还支持用number[1..2]来表示原 集合的子集合[“2”“3”]

    3. 输出Map元素
    对于JavaBean实例,FreeMarker一样把它看作属性为key,属性值为valueMap对象。
    输出Map对象时,可以使用点语法或中括号语法,如下面的几种写法的效果是一样的:
                 book.author.name                                                                                                                        
                 book.author["name"]                                                                                                                     
                 book["author"].name                                                                                                                     
                 book["author"]["name"]                                                                                                                  
    使用点语法时,变量名字有和顶层变量一样的限制,但中括号语法没有任何限制。

    三、字符串操作

    1. 字符串连接
    字符串连接有两种语法:
    1) 使用${..}#{..}在字符串常量内插入表达式的值;
    (2)  直接使用连接运算符“+”连接字符串。
    如,下面两种写法等效:
                  ${"Hello, ${user}"}                                                                                                               
                  ${"Hello, " + user + "!"}                                                                                                        
    有一点需要注意: ${..}只能用于文本部分作为插值输出,而不能用于比较等其他用途,如:
                  <#if ${isBig}>Wow!</#if>                                                                                                               
                  <#if "${isBig}">Wow!</#if>                                                                                                             
    应该写成:
                  <#if isBig>Wow!</#if>                                                                                                                    

    2. 截取子串
    截取子串可以根据字符串的索引来进行,如果指定一个索引值,则取得字符串该索引处的字符;如果指定两个索引值,则截取两个索引中间的字符串子串。如:
                  <#assign number="01234">
                  ${number[0]} <#-- 输出字符0 -->
                  ${number[0..3]} <#-- 输出子串“0123” -->

     

    四、集合连接操作
          连接集合的运算符为“+”

    五、Map连接操作
       Map连接操作的运算符为“+”

    六、算术运算符
       FreeMarker表达式中支持“+”“*”“/”“%”运算符。

    七、比较运算符

    表达式中支持的比较运算符有如下几种:
    1. =(或者==): 判断两个值是否相等;
    2. !=: 判断两个值是否不相等;
    注: =!=可以用作字符串、数值和日期的比较,但两边的数据类型必须相同。而且FreeMarker的比较是精确比较,不会忽略大小写及空格。
    3. >(或者gt): 大于
    4. >=(或者gte): 大于等于
    5. <(或者lt): 小于
    6. <=(或者lte): 小于等于
    注: 上面这些比较运算符可以用于数字和日期,但不能用于字符串。大部分时候,使用gt>有更好的效果,因为FreeMarker会把>解释成标签的结束字符。可以使用括号来避免这种情况,如:<#if (x>y)>

    if else 语句测试:

    <#if num0 gt 18>  <#--不是使用>,大部分时候,freemarker会把>解释成标签结束! -->

    及格!

    <#else>

    不及格!

    </#if>

    root.put("num0", 18);

    八、逻辑运算符

    1. &&: 逻辑与;
    2. ||: 逻辑或;
    3. !: 逻辑非
    逻辑运算符只能用于布尔值。

    九、内建函数

    FreeMarker提供了一些内建函数来转换输出,可以在任何变量后紧跟??后紧跟内建函数,就可以通过内建函数来转换输出变量。

    字符串相关常用的内建函数:
    1. html: 对字符串进行HTML编码;
    2. cap_first: 使字符串第一个字母大写;
    3. lower_case: 将字符串转成小写;
    4. upper_case: 将字符串转成大写;

    集合相关常用的内建函数:
    1. size: 获得集合中元素的个数;

    数字值相关常用的内建函数:
    1. int: 取得数字的整数部分。

    举例:

    root.put("htm2", "<b>粗体</b>");

    内建函数:

    ${htm2?html}

    十、空值处理运算符

    FreeMarker的变量必须赋值,否则就会抛出异常。而对于FreeMarker来说,null值和不存在的变量是完全一样的,因为FreeMarker无法理解null值。
    FreeMarker提供两个运算符来避免空值:
    1. !: 指定缺失变量的默认值;
    2. ??:判断变量是否存在。
    !运算符有两种用法:variable!variable!defaultValue。第一种用法不给变量指定默认值,表明默认值是空字符串、长度为0的集合、或长度为0Map对象。
    使用!运算符指定默认值并不要求默认值的类型和变量类型相同。

    测试空值处理:

    <#-- ${sss} 没有定义这个变量,会报异常! -->

    ${sss!} <#--没有定义这个变量,默认值是空字符串! -->

    ${sss!"abc"} <#--没有定义这个变量,默认值是字符串abc-->

    ??运算符返回布尔值,如:variable??,如果变量存在,返回true,否则返回false

    模板开发语句

    最简单的模板是普通  HTML  文件(或者是其他任何文本文件—FreeMarker  本身不属于HTML)。当客户端访问页面时,FreeMarker 要发送 HTML 代码至客户端浏览器端显示。如果想要页面动起来,就要在 HTML 中放置能被 FreeMarker 所解析的特殊部分。

    ${}FreeMarker 将会输出真实的值来替换花括号内的表达式,这样的表达式被称为

    interpolations 插值,可以参考第上面示例的内容。

    FTL tags 标签(FreeMarker  模板的语言标签):FTL 标签和 HTML 标签有一点相似,但是它们是  FreeMarker  的指令而且是不会直接输出出来的东西。这些标签的使用一般以符号#开头。(用户自定义的 FTL 标签使用@符号来代替#,但这是更高级的主题内容了,后面会详细地讨论)

    Comments 注释:FreeMarker 的注释和 HTML 的注释相似,但是它用<#---->来分隔的。任何介于这两个分隔符(包含分隔符本身)之间内容会被 FreeMarker  忽略,就不会

    输出出来了。

    其他任何不是  FTL  标签,插值或注释的内容将被视为静态文本,这些东西就不会被FreeMarker 所解析,会被按照原样输出出来。

    directives指令:就是所指的FTL  标签。这些指令在HTML  的标签(如<table></table>)和 HTML 元素(如 table 元素)中的关系是相同的。(如果现在你还不能区分它们,那么把FTL 标签”和“指令”看做是同义词即可。)

    if指令

    root.put("random", new Random().nextInt(100));

    ------------------------------------------------

    if语句测试:

    ${user}<#if user=="老高">我们的老师</#if>

    ------------------------------------------------

    if else 语句测试:

    <#if num0 gt 18>  <#--不是使用>,大部分时候,freemarker会把>解释成标签结束! -->

    及格!

    <#else>

    不及格!

    </#if>

    ---------------------------------------------------

    if else if else语句测试:

    <#if random gte 90>

    优秀!

    <#elseif random gte 80>

    良好!

    <#else>

    一般!

    </#if>

    ----------------------------------------------------

    list指令

    list指令是一个迭代输出指令,用于迭代输出数据模型中的集合,list指令的语法格式如下

    1 <#list sequence as item> 
    2 ... 
    3 </#list> 

    上面的语法格式中,sequence就是一个集合对象,也可以是一个表达式,但该表达式将返回一个集合对象,item是一个任意的名字,就是被迭代输出的集合元素.此外,迭代集合对象时,还包含两个特殊的循环变量
    item_index:当前变量的索引值 
    item_has_next:是否存在下一个对象 
    也可以使用<#break>指令跳出迭代 
    例子如下

    1 <#list ["星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期天"] as x> 
    2 ${x_index + 1}.${x}<#if x_has_next>,</if> 
    3 <#if x="星期四"><#break></#if> 
    4 </#list> 

    switch , case , default , break指令 
    这些指令显然是分支指令,作用类似于Javaswitch语句,switch指令的语法结构如下

    1 <#switch value> 
    2 <#case refValue>...<#break> 
    3 <#case refValue>...<#break> 
    4 <#default>... 
    5 </#switch> 

    include指令 
    include指令的作用类似于JSP的包含指令,用于包含指定页.include指令的语法格式如下
    <#include filename [options]> 
    在上面的语法格式中,两个参数的解释如下
    filename:该参数指定被包含的模板文件 
    options:该参数可以省略,指定包含时的选项,包含encodingparse两个选项,其中encoding指定包含页面时所用的解码集,parse指定被包含文件是否作为FTL文件来解析,如果省略了parse选项值,则该选项默认是true. 

    import指令 
    该指令用于导入FreeMarker模板中的所有变量,并将该变量放置在指定的Map对象中,import指令的语法格式如下
    <#import "/lib/common.ftl" as com> 
    上面的代码将导入/lib/common.ftl模板文件中的所有变量,交将这些变量放置在一个名为comMap对象中

    noparse指令 
    noparse指令指定FreeMarker不处理该指定里包含的内容,该指令的语法格式如下
    <#noparse>...</#noparse> 
    看如下的例子

    1 <#noparse> 
    2 <#list books as book> 
    3    <tr><td>${book.name}<td>作者:${book.author} 
    4 </#list> 
    5 </#noparse> 

    输出如下

    1 <#list books as book> 
    2    <tr><td>${book.name}<td>作者:${book.author} 
    3 </#list> 

    assign指令

    在前面已经使用了多次,它用于为该模板页面创建或替换一个顶层变量,assign指令的用法有多种,包含创建或替换一个顶层变量, 或者创建或替换多个变量等,它的最简单的语法如下:<#assign name=value [in namespacehash]>,这个用法用于指定一个名为name的变量,该变量的值为value,此外,FreeMarker允许在使用 assign指令里增加in子句,in子句用于将创建的name变量放入namespacehash命名空间中
    assign指令还有如下用法:<#assign name1=value1 name2=value2 ... nameN=valueN [in namespacehash]>,这个语法可以同时创建或替换多个顶层变量,此外,还有一种复杂的用法,如果需要创建或替换的变量值是一个复杂的表达式,则可以使用如下语法格式:<#assign name [in namespacehash]>capture this</#assign>,在这个语法中,是指将assign指令的内容赋值给name变量.如下例子

    1 <#assign x> 
    2 <#list ["星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期天"] as n> 
    3 ${n} 
    4 </#list> 
    5 </#assign> 
    6 ${x} 

    上面的代码将产生如下输出:星期一 星期二 星期三 星期四 星期五 星期六 星期天 

     

     setting指令 
    该指令用于设置FreeMarker的运行环境,该指令的语法格式如下:<#setting name=value>,在这个格式中,name的取值范围包含如下几个
    locale:该选项指定该模板所用的国家/语言选项 
    number_format:指定格式化输出数字的格式 
    boolean_format:指定两个布尔值的语法格式,默认值是true,false 
    date_format,time_format,datetime_format:指定格式化输出日期的格式 
    time_zone:设置格式化输出日期时所使用的时区 

     

    return指令

    用于结束macro指令,一旦在macro指令中执行了return指令,FreeMarker不会继续处理macro指令里的内容,看如下代码
    <#macro book> 
    spring 
    <#return> 
    j2ee 
    </#macro> 
    <@book /> 
    上面的代码输出:spring,j2ee位于return指令之后,不会输出

    macro指令

    可以用于实现自定义指令,通过使用自定义指令,可以将一段模板片段定义成一个用户指令,使用macro指令的语法格式如下

    1 <#macro name param1 param2 ... paramN> 
    2 ... 
    3 <#nested loopvar1, loopvar2, ..., loopvarN> 
    4 ... 
    5 <#return> 
    6 ... 
    7 </#macro> 
    1 <#macro book>   //定义一个自定义指令 
    2 j2ee 
    3 </#macro> 
    4 <@book />       //使用刚才定义的指令 
    5 上面的代码输出结果为:j2ee 
  • 相关阅读:
    [Tutorial] The find command used in linux terminal ...
    Python: An example of plotting bar charts using python
    [Installation] Install Matlab2016b on Ubuntu 14.04
    [Debug] Fixed the catkin_make broken problem after intalling newer Python with Anaconda
    [Installation] My Workbench under construction
    [Installation] Install Python 3.5.2 from source
    实操 | Getting started with Self-Driving Car : Prepare
    Python: A python tools: iPython Notebook
    路线图 | Getting started with Tensorflow
    实操 | Building a ROS simulation platform for Deep Reinforcement Learning research ...
  • 原文地址:https://www.cnblogs.com/jiaqingshareing/p/7484549.html
Copyright © 2011-2022 走看看