国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

工具集核心教程 | 第六篇: Freemarker模板引擎入門到進階

趙連江 / 1813人閱讀

摘要:到目前為止,使用越來越廣泛,不光光只是它強大的生成技術,而且它能夠與進行很好的集成。注意使用數字范圍來定義集合時無需使用方括號數字范圍也支持反遞增的數字范圍如對象對象使用花括號包括中的對之間以英文冒號分隔,多組對之間以英文逗號分隔。

Freemarker的介紹

??Freemarker 是一款模板引擎,是一種基于模版生成靜態文件的通用 工具,它是為程序員提供的一個開發包,或者說是一個類庫,它不是面向最終用戶的,而是為程序員提供了一款可以嵌入他們開發產品的應用程序。
??Freemarker 是使用純java編寫的,為了提高頁面的訪問速度,需要把頁面靜態化, 那么Freemarker就是被用來生成html頁面。
??到目前為止,Freemarker使用越來越廣泛,不光光只是它強大的生成技術,而且它能夠與進行很好的集成。
?現在開始一層層揭開它的神秘面紗。。。

特點
1.  輕量級模版引擎,不需要Servlet環境就可以很輕松的嵌入到應用程序中
2.  能生成各種文本,如html,xml,java,等
3.  入門簡單,它是用java編寫的,很多語法和java相似

工作原理:(借用網上的圖片)

目錄
1.FTL指令規則
2.插值規則
3.表達式
4.FreeMarker的常用指令
5.高級方法
前言

FreeMarker的模板文件并不比HTML頁面復雜多少,FreeMarker模板文件主要由如下4個部分組成:

1.文本:直接輸出的部分
2.注釋:<#-- ... -->格式部分,不會輸出
3.插值:即${...}或#{...}格式的部分,將使用數據模型中的部分替代輸出
4.FTL指令:FreeMarker指定,和HTML標記類似,名字前加#予以區分,不會輸出

下面是一個FreeMarker模板的例子,包含了以上所說的4個部分:


    
        Welcome!
    
    
        <#-- 注釋部分 -->
        <#-- 下面使用插值 -->
        

Welcome ${user} !

We have these animals:

<#-- 使用FTL指令 --> <#list animals as being>
  • ${being.name} for ${being.price} Euros
  • <#list>
    1. FTL指令規則

    FreeMarker中,使用FTL標簽來使用指令,FreeMarker3FTL標簽,這和HTML標簽是完全類似的.

    開始標簽:<#directivename parameter>
    結束標簽:
    空標簽:<#directivename parameter/>
    

    實際上,使用標簽時前面的符號#也可能變成@,如果該指令是一個用戶指令而不是系統內建指令時,應將#符號改成@符號.

    使用FTL標簽時,應該有正確的嵌套,而不是交叉使用,這和XML標簽的用法完全一樣.如果全用不存在的指令,FreeMarker不會使用模板輸出,而是產生一個錯誤消息.FreeMarker會忽略FTL標簽中的空白字符.值得注意的是< , /> 和指令之間不允許有空白字符.

    2. 插值規則

    FreeMarker的插值有如下兩種類型:

    通用插值${expr};
    數字格式化插值:#{expr}或#{expr;format}
    
    2.1 通用插值

    對于通用插值,又可以分為以下4種情況:

    1.插值結果為字符串值:直接輸出表達式結果
    2.插值結果為數字值:根據默認格式(由#setting指令設置)將表達式結果轉換成文本輸出.可以使用內建的字符串   函數格式化單個插值,如下面的例子:
    <#settion number_format="currency"/>
    <#assign answer=42/>
    ${answer}
    ${answer?string} <#-- the same as ${answer} -->
    ${answer?string.number}
    ${answer?string.currency}
    ${answer?string.percent}
    ${answer}

    輸出結果是:

    $42.00
    $42.00
    42
    $42.00
    4,200%
    3.插值結果為日期值:根據默認格式(由#setting指令設置)將表達式結果轉換成文本輸出.可以使用內建的字符串函數格式化單個插值,如下面的例子:
    ${lastUpdated?string("yyyy-MM-dd HH:mm:ss zzzz")}
    ${lastUpdated?string("EEE, MMM d, ""yy")}
    ${lastUpdated?string("EEEE, MMMM dd, yyyy, hh:mm:ss a "("zzz")"")}

    輸出結果是:

    2008-04-08 08:08:08 Pacific Daylight Time
    Tue, Apr 8, "03
    Tuesday, April 08, 2003, 08:08:08 PM (PDT)
    4.插值結果為布爾值:根據默認格式(由#setting指令設置)將表達式結果轉換成文本輸出.可以使用內建的字符串函數格式化單個插值,如下面的例子:
    <#assign foo=true/>
    ${foo?string("yes", "no")}

    輸出結果是:

    yes
    2.2 數字格式化插值

    數字格式化插值可采用#{expr;format}形式來格式化數字,其中format可以是:

    mX:小數部分最小X位
    MX:小數部分最大X位
    

    如下面的例子:

    <#assign x=2.582/>
    <#assign y=4/>
    #{x; M2} <#-- 輸出2.58 -->
    #{y; M2} <#-- 輸出4 -->
    #{x; m2} <#-- 輸出2.6 -->
    #{y; m2} <#-- 輸出4.0 -->
    #{x; m1M2} <#-- 輸出2.58 -->
    #{x; m1M2} <#-- 輸出4.0 -->
    3. 表達式

    表達式是FreeMarker模板的核心功能,表達式放置在插值語法${}之中時,表明需要輸出表達式的值;表達式語法也可與FreeMarker標簽結合,用于控制輸出.實際上FreeMarker的表達式功能非常強大,它不僅支持直接指定值,輸出變量值,也支持字符串格式化輸出和集合訪問等功能.

    3.1 直接指定值

    使用直接指定值語法讓FreeMarker直接輸出插值中的值,而不是輸出變量值.直接指定值可以是字符串,數值,布爾值,集合和MAP對象.

    字符串
    

    直接指定字符串值使用單引號或雙引號限定,如果字符串值中包含特殊字符需要轉義,看下面的例子:

    ${"我的文件保存在C:盤"}
    ${"我名字是"annlee""}

    輸出結果是:

    我的文件保存在C:盤
    我名字是"annlee"

    FreeMarker支持如下轉義字符:

    "; 雙引號(u0022)
    "; 單引號(u0027)
    ; 反斜杠(u005C)
    
    ; 換行(u000A)
    
    ; 回車(u000D)
    	; Tab(u0009)
    ; 退格鍵(u0008)
    f; Form feed(u000C)
    l; <
    g; >
    a; &
    {; {
    xCode; 直接通過4位的16進制數來指定Unicode碼,輸出該unicode碼對應的字符.

    如果某段文本中包含大量的特殊符號,FreeMarker提供了另一種特殊格式:可以在指定字符串內容的引號前增加r標記,在r標記后的文件將會直接輸出。看如下代碼:

    ${r"${foo}"}
    ${r"C:fooar"}

    輸出結果是:

    ${foo}
    C:fooar
    數值
    

    表達式中的數值直接輸出,不需要引號.小數點使用"."分隔,不能使用分組","符號.FreeMarker目前還不支持科學計數法,所以"1E3"是錯誤的.在FreeMarker表達式中使用數值需要注意以下幾點:

    數值不能省略小數點前面的0,所以".5"是錯誤的寫法
    
    數值8 , +8 , 8.00都是相同的
    
    布爾值,直接使用true和false,不使用引號.
    
    集合, 集合以方括號包括,各集合元素之間以英文逗號","分隔
    
    如下為集合元素遍歷的例子:
    
    <#list ["星期一", "星期二", "星期三", "星期四", "星期五",     "星期六", "星期天"] as x>
        ${x}
    
    
    輸出結果是:
    
    星期一
    星期二
    星期三
    星期四
    星期五
    星期六
    星期天
    
    除此之外,集合元素也可以是表達式,例子如下:
    
    [2 + 2, [1, 2, 3, 4], "whatnot"]
    
    還可以使用數字范圍定義數字集合,如2..5等同于[2, 3, 4, 5], 但是更有效率。
    
        注意,使用數字范圍來定義集合時無需使用方括號,數字范圍也支持反遞增的數字范圍,如5..2
    
    Map對象,Map對象使用花括號包括,Map中的key-value對之間以英文冒號":"分隔,多組key-value對之間以英文逗號","分隔。
    
    下面是一個例子:
    
    {"語文":78, "數學":80}
    
    Map對象的key和value都是表達式,但是key必須是字符串。
    
    3.2 輸出變量值

    FreeMarker的表達式輸出變量時,這些變量可以是頂層變量,也可以是Map對象中的變量,還可以是集合中的變量,并可以使用點(.)語法來訪問Java對象的屬性。下面分別討論這些情況:

    1. 頂層變量

    所謂頂層變量就是直接放在數據模型中的值,例如有如下數據模型:

    Map root = new HashMap();   //創建數據模型
    root.put("name","annlee");   //name是一個頂層變量

    對于頂層變量,直接使用${variableName}來輸出變量值,變量名只能是字母,數字,下劃線,$,@#的組合,且不能以數字開頭號.為了輸出上面的name的值,可以使用如下語法:

    ${name}
    
    2. 輸出集合元素

    如果需要輸出集合元素,則可以根據集合元素的索引來輸出集合元素,集合元素的索引以方括號指定。

    假設有索引:["星期一","星期二","星期三","星期四","星期五","星期六","星期天"],該索引名為week,如果需要輸出星期三,則可以使用如下語法:

    ${week[2]} //輸出第三個集合元素

    此外,FreeMarker還支持返回集合的子集合,如果需要返回集合的子集合,則可以使用如下語法:

    week[3..5] //返回week集合的子集合,子集合中的元素是week集合中的第4-6個元素
    3. 輸出Map元素

    這里的Map對象可以是直接HashMap的實例,甚至包括JavaBean實例,對于JavaBean實例而言,我們一樣可以把其當成屬性為key,屬性值為valueMap實例.為了輸出Map元素的值,可以使用點語法或方括號語法.假如有下面的數據模型:

    Map root = new HashMap();
    Book book = new Book();
    Author author = new Author();
    author.setName("annlee");
    author.setAddress("gz");
    book.setName("struts2");
    book.setAuthor(author);
    root.put("info","struts");
    root.put("book", book);

    為了訪問數據模型中名為struts2的書的作者的名字,可以使用如下語法:

    book.author.name //全部使用點語法
    book["author"].name
    book.author["name"] //混合使用點語法和方括號語法
    book["author"]["name"] //全部使用方括號語法

    使用點語法時,變量名字有頂層變量一樣的限制,但方括號語法沒有該限制,因為名字可以是任意表達式的結果.

    3.3 字符串操作

    FreeMarker的表達式對字符串操作非常靈活,可以將字符串常量和變量連接起來,也可以返回字符串的子串等。

    字符串連接有兩種語法:

    使用${..}或#{..}在字符串常量部分插入表達式的值,從而完成字符串連接.
    直接使用連接運算符+來連接字符串
    

    例如有如下數據模型:

    Map root = new HashMap(); 
    root.put("user","annlee");

    下面將user變量和常量連接起來:

    ${"hello, ${user}!"}   //使用第一種語法來連接
    ${"hello, " + user + "!"} //使用+號來連接

    上面的輸出字符串都是hello,annlee!,可以看出這兩種語法的效果完全一樣.
    值得注意的是,${..}只能用于文本部分,不能用于表達式,下面的代碼是錯誤的:

    <#if ${isBig}>Wow!
    <#if "${isBig}">Wow!

    應該寫成:

    <#if isBig>Wow!

    截取子串可以根據字符串的索引來進行,截取子串時如果只指定了一個索引值,則用于取得字符串中指定索引所對應的字符;如果指定兩個索引值,則返回兩個索引中間的字符串子串。假如有如下數據模型:

    Map root = new HashMap(); 
    root.put("book","struts2,freemarker");

    可以通過如下語法來截取子串:

    ${book[0]}${book[4]}   //結果是su
    ${book[1..4]}     //結果是tru
    3.4 集合連接運算符

    這里所說的集合運算符是將兩個集合連接成一個新的集合,連接集合的運算符是+,看如下的例子:

    <#list ["星期一","星期二","星期三"] + ["星期四","星期五","星期六","星期天"] as x>
        ${x}
    

    輸出結果是:

    星期一 星期二 星期三 星期四 星期五 星期六 星期天
    3.5 Map連接運算符

    Map對象的連接運算符也是將兩個Map對象連接成一個新的Map對象,Map對象的連接運算符是+,如果兩個Map對象具有相同的key,則右邊的值替代左邊的值.看如下的例子:

    <#assign scores = {"語文":86,"數學":78} + {"數學":87,"Java":93}>
    語文成績是${scores.語文}
    數學成績是${scores.數學}
    Java成績是${scores.Java}

    輸出結果是:

    語文成績是86
    數學成績是87
    Java成績是93
    3.6 算術運算符

    FreeMarker表達式中完全支持算術運算,FreeMarker支持的算術運算符包括:+, - , * , / , % 看如下的代碼:

    <#assign x=5>
    ${ x * x - 100 }
    ${ x /2 }
    ${ 12 %10 }

    輸出結果是:

    -75   2.5   2

    在表達式中使用算術運算符時要注意以下幾點:

    運算符兩邊的運算數字必須是數字
    使用+運算符時,如果一邊是數字,一邊是字符串,就會自動將數字轉換為字符串再連接,如: ${3 + "5"},結果是:35
    

    使用內建的int函數可對數值取整,如:

    <#assign x=5>
    ${ (x/2)?int }
    ${ 1.1?int }
    ${ 1.999?int }
    ${ -1.1?int }
    ${ -1.999?int }

    結果是:

    2 1 1 -1 -1
    3.7 比較運算符

    表達式中支持的比較運算符有如下幾個:

    =或者==:判斷兩個值是否相等.
    !=:判斷兩個值是否不等.
    >或者gt:判斷左邊值是否大于右邊值
    >=或者gte:判斷左邊值是否大于等于右邊值
    <或者lt:判斷左邊值是否小于右邊值
    <=或者lte:判斷左邊值是否小于等于右邊值
    
    注意:=和!=可以用于字符串,數值和日期來比較是否相等,但=和!=兩邊必須是相同類型的值,否則會產生錯誤,而且FreeMarker是精確比較,"x","x ","X"是不等的.其它的運行符可以作用于數字和日期,但不能作用于字符串,大部分的時候,使用gt等字母運算符代替>會有更好的效果,因為FreeMarker會把>解釋成FTL標簽的結束字符,當然,也可以使用括號來避免這種情況,如:<#if (x>y)>。
    
    3.8 邏輯運算符

    邏輯運算符有如下幾個:

    邏輯與: &&
    邏輯或: ||
    邏輯非: !

    邏輯運算符只能作用于布爾值,否則將產生錯誤。

    3.9 內建函數

    FreeMarker還提供了一些內建函數來轉換輸出,可以在任何變量后緊跟“?”“?”后緊跟內建函數,就可以通過內建函數來輪換輸出變量.下面是常用的內建的字符串函數:

    html:對字符串進行HTML編碼
    cap_first:使字符串第一個字母大寫
    lower_case:將字符串轉換成小寫
    upper_case:將字符串轉換成大寫
    trim:去掉字符串前后的空白字符

    下面是集合的常用內建函數

    size:獲取序列中元素的個數
    

    下面是數字值的常用內建函數

    int:取得數字的整數部分,結果帶符號
    

    例如:

    <#assign test="Tom & Jerry">
    ${test?html}
    ${test?upper_case?html}

    結果是:

    Tom & Jerry   TOM & JERRY
    3.10 空值處理運算符

    FreeMarker對空值的處理非常嚴格,FreeMarker的變量必須有值,沒有被賦值的變量就會拋出異常,因為FreeMarker未賦值的變量強制出錯可以杜絕很多潛在的錯誤,如缺失潛在的變量命名,或者其他變量錯誤.這里所說的空值,實際上也包括那些并不存在的變量,對于一個Javanull值而言,我們認為這個變量是存在的,只是它的值為null,但對于FreeMarker模板而言,它無法理解null值,null值和不存在的變量完全相同.

    為了處理缺失變量,FreeMarker提供了兩個運算符:

    ! :指定缺失變量的默認值
    ??:判斷某個變量是否存在
    

    其中,!運算符的用法有如下兩種:variable!variable!defaultValue,第一種用法不給缺失的變量指定默認值,表明默認值是空字符串,長度為0的集合,或者長度為0Map對象。使用!指定默認值時,并不要求默認值的類型和變量類型相同.使用??運算符非常簡單,它總是返回一個布爾值,用法為:variable??,如果該變量存在,返回true,否則返回false

    3.11 運算符的優先級

    FreeMarker中的運算符優先級如下(由高到低排列):

    一元運算符: !
    內建函數: ?
    乘除法: *, / , %
    加減法: - , +
    比較: > , < , >= , <= (lt , lte , gt , gte)
    相等: == , = , !=
    邏輯與: &&
    邏輯或: ||
    數字范圍: ..
    

    實際上,我們在開發過程中應該使用括號來嚴格區分,這樣的可讀性好,出錯少。

    4. FreeMarker的常用指令

    FreeMarkerFTL指令也是模板的重要組成部分,這些指令可實現對數據模型所包含數據的撫今迭代,分支控制.除此之外,還有一些重要的功能,也是通過FTL指令來實現的.

    4.1 if指令

    這是一個典型的分支控制指令,該指令的作用完全類似于Java語言中的if,if指令的語法格式如下:

    <#if condition>...
    <#elseif condition>...
    <#elseif condition>...
    <#else> ...
    

    例子如下:

    <#assign age=23>
    <#if (age>60)>
        老年人
    <#elseif (age>40)>
        中年人
    <#elseif (age>20)>
        青年人
    <#else> 
        少年人
    

    輸出結果是:

    青年人

    上面的代碼中的邏輯表達式用括號括起來主要是因為里面有>符號,由于FreeMarker會將>符號當成標簽的結束字符,可能導致程序出錯,為了避免這種情況,我們應該在凡是出現這些符號的地方都使用括號。

    4.2 switch , case , default , break指令

    這些指令顯然是分支指令,作用類似于Javaswitch語句,switch指令的語法結構如下:

    <#switch value>
        <#case refValue>...<#break>
        <#case refValue>...<#break>
        <#default>...
    
    4.3 list, break指令

    list指令是一個迭代輸出指令,用于迭代輸出數據模型中的集合,list指令的語法格式如下:

    <#list sequence as item>
        ...
    

    上面的語法格式中,sequence就是一個集合對象,也可以是一個表達式,但該表達式將返回一個集合對象,而item是一個任意的名字,就是被迭代輸出的集合元素.此外,迭代集合對象時,還包含兩個特殊的循環變量:

    item_index:當前變量的索引值
    item_has_next:是否存在下一個對象
    

    除此之外,也可以使用<#break>指令跳出迭代。

    例子如下:

    <#list ["星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期天"] as x>
        ${x_index + 1}.${x}<#if x_has_next>,
        <#if x="星期四"><#break>
    
    4.4 include指令

    include指令的作用類似于JSP的包含指令,用于包含指定頁.include指令的語法格式如下:

    <#include filename [options]>
    

    在上面的語法格式中,兩個參數的解釋如下:

    filename:該參數指定被包含的模板文件
    options:該參數可以省略,指定包含時的選項,包含encoding和parse兩個選項,其中encoding指定包含頁面時所用的解碼集,而parse指定被包含文件是否作為FTL文件來解析,如果省略了parse選項值,則該選項默認是true.
    
    4.5 import指令

    import指令用于導入FreeMarker模板中的所有變量,并將該變量放置在指定的Map對象中,import指令的語法格式如下:

    <#import "/lib/common.ftl" as com>
    

    上面的代碼將導入/lib/common.ftl模板文件中的所有變量,交將這些變量放置在一個名為comMap對象中.

    4.6 noparse指令

    noparse指令指定FreeMarker不處理該指定里包含的內容,該指令的語法格式如下:

    <#noparse>...
    

    看如下的例子:

    <#noparse>
    <#list books as book>
       ${book.name}作者:${book.author}
    
    

    輸出如下:

    <#list books as book>
       ${book.name}作者:${book.author}
    
    4.7 escape , noescape指令

    escape指令導致body區的插值都會被自動加上escape表達式,但不會影響字符串內的插值,只會影響到body內出現的插值,使用escape指令的語法格式如下:

    <#escape identifier as expression>...
        <#noescape>...
    

    看如下的代碼:

    <#escape x as x?html>
        First name:${firstName}
        Last name:${lastName}
        Maiden name:${maidenName}
    

    上面的代碼等同于:

    First name:${firstName?html}
    Last name:${lastName?html}
    Maiden name:${maidenName?html}

    escape指令在解析模板時起作用而不是在運行時起作用,除此之外,escape指令也嵌套使用,子escape繼承父escape的規則,如下例子:

    <#escape x as x?html>
        Customer Name:${customerName}
        Items to ship;
        <#escape x as itemCodeToNameMap[x]>
           ${itemCode1}
           ${itemCode2}
           ${itemCode3}
           ${itemCode4}
        
    

    上面的代碼類似于:

    Customer Name:${customerName?html}
    Items to ship;
    ${itemCodeToNameMap[itemCode1]?html}
    ${itemCodeToNameMap[itemCode2]?html}
    ${itemCodeToNameMap[itemCode3]?html}
    ${itemCodeToNameMap[itemCode4]?html}

    對于放在escape指令中所有的插值而言,這此插值將被自動加上escape表達式,如果需要指定escape指令中某些插值無需添加escape表達式,則應該使用noescape指令,放在noescape指令中的插值將不會添加escape表達式。

    4.8 assign指令

    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指令的內容賦值給name變量。如下例子:

    <#assign x>
        <#list ["星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期天"] as n>
            ${n}
        
    
    ${x}

    上面的代碼將產生如下輸出:

    星期一 星期二 星期三 星期四 星期五 星期六 星期天

    雖然assign指定了這種復雜變量值的用法,但是我們也不要濫用這種用法,如下例子:

    <#assign x>Hello ${user}!

    以上代碼改為如下寫法更合適:

    <#assign x="Hello ${user}!">
    4.9 setting指令

    該指令用于設置FreeMarker的運行環境,該指令的語法格式如下:<#setting name=value>,在這個格式中,name的取值范圍包含如下幾個:

    locale:該選項指定該模板所用的國家/語言選項
    number_format:指定格式化輸出數字的格式
    boolean_format:指定兩個布爾值的語法格式,默認值是true,false
    date_format,time_format,datetime_format:指定格式化輸出日期的格式
    time_zone:設置格式化輸出日期時所使用的時區
    
    4.10 macro , nested , return指令

    macro可以用于實現自定義指令,通過使用自定義指令,可以將一段模板片段定義成一個用戶指令,使用macro指令的語法格式如下:

    <#macro name param1 param2 ... paramN>
        ...
        <#nested loopvar1, loopvar2, ..., loopvarN>
        ...
        <#return>
        ...
    

    在上面的格式片段中,包含了如下幾個部分:

    name:name屬性指定的是該自定義指令的名字,使用自定義指令時可以傳入多個參數
    paramX:該屬性就是指定使用自定義指令時報參數,使用該自定義指令時,必須為這些參數傳入值
    nested指令:nested標簽輸出使用自定義指令時的中間部分
    nested指令中的循環變量:這此循環變量將由macro定義部分指定,傳給使用標簽的模板
    return指令:該指令可用于隨時結束該自定義指令.
    

    看如下的例子:

    <#macro book>   //定義一個自定義指令
    j2ee
    
    <@book />    //使用剛才定義的指令

    上面的代碼輸出結果為:

    j2ee

    在上面的代碼中,可能很難看出自定義標簽的用處,因為我們定義的book指令所包含的內容非常簡單,實際上,自定義標簽可包含非常多的內容,從而可以實現更好的代碼復用.此外,還可以在定義自定義指令時,為自定義指令指定參數,看如下代碼:

    <#macro book booklist>     //定義一個自定義指令booklist是參數
        <#list booklist as book>
           ${book}
        
    
    <@book booklist=["spring","j2ee"] />   //使用剛剛定義的指令

    上面的代碼為book指令傳入了一個參數值,上面的代碼的輸出結果為:

    spring j2ee

    不僅如此,還可以在自定義指令時使用nested指令來輸出自定義指令的中間部分,看如下例子:

    <#macro page title>
    
    
       FreeMarker示例頁面 - ${title?html}
    
    
       

    ${title?html}

    <#nested> //用于引入用戶自定義指令的標簽體

    上面的代碼將一個HTML頁面模板定義成一個page指令,則可以在其他頁面中如此page指令:

    <#import "/common.ftl" as com>     //假設上面的模板頁面名為common.ftl,導入頁面
    <@com.page title="book list">
    
    
  • spring
  • j2ee
  • 從上面的例子可以看出,使用macronested指令可以非常容易地實現頁面裝飾效果,此外,還可以在使用nested指令時,指定一個或多個循環變量,看如下代碼:

    <#macro book>
    <#nested 1>      //使用book指令時指定了一個循環變量值
    <#nested 2>
    
    <@book ;x> ${x} .圖書

    當使用nested指令傳入變量值時,在使用該自定義指令時,就需要使用一個占位符(如book指令后的;x).上面的代碼輸出文本如下:

    1 .圖書    2 .圖書

    nested指令中使用循環變量時,可以使用多個循環變量,看如下代碼:

    <#macro repeat count>
        <#list 1..count as x>     //使用nested指令時指定了三個循環變量
           <#nested x, x/2, x==count>
        
    
    <@repeat count=4 ; c halfc last>
        ${c}. ${halfc}<#if last> Last! 
    

    上面的輸出結果為:

    1. 0.5   2. 1   3. 1.5   4. 2 Last;

    return指令用于結束macro指令,一旦在macro指令中執行了return指令,則FreeMarker不會繼續處理macro指令里的內容,看如下代碼:

    <#macro book>
    spring
    <#return>
    j2ee
    
    <@book />

    上面的代碼輸出:

    spring

    j2ee位于return指令之后,不會輸出。

    4.11 t, lt, rt 指令

    語法:

    <#t> 去掉左右空白和回車換行
    <#lt> 去掉左邊空白和回車換行
    <#rt> 去掉右邊空白和回車換行
    <#nt> 取消上面的效果
    4.12 指令總結
    if, else, elseif
    switch, case, default, break
    list, break
    include
    Import
    compress
    escape, noescape
    assign
    global
    setting
    macro, nested, return
    t, lt, rt, nt
    
    5. 高級方法 5.1 自定義方法

    自定義方法, 如:

    ${timer("yyyy-MM-dd H:mm:ss", x)}
    ${timer("yyyy-MM-dd ", x)}

    在模板中除了可以通過對象來調用方法外(${object.methed(args)})也可以直接調用java實現的方法,java類必須實現接口TemplateMethodModel的方法exec(List args). 下面以把毫秒的時間轉換成按格式輸出的時間為例子:

    public class LongToDate implements TemplateMethodModel {
        
        public TemplateModel exec(List args) throws TemplateModelException {
            SimpleDateFormat mydate = new SimpleDateFormat((String) args.get(0)));
            return mydate.format(new Date(Long.parseLong((String)args.get(1)));
        }
    }  

    LongToDate對象放入到數據模型中:

        root.put("timer", new LongToDate());

    ftl模板里使用:

    <#assign x = "123112455445">
    ${timer("yyyy-MM-dd H:mm:ss", x)}
    ${timer("yyyy-MM-dd ", x)}

    輸出結果:

    2001-10-12 5:21:12
    2001-10-12
    5.2 自定義 Transforms

    實現自定義的<@transform>文本或表達式的功能,允許對中間的最終文本進行解析轉換

    例子:實現<@upcase>strstr轉換成STR的功能.

    代碼如下:

    import java.io.*;
    import java.util.*;
    import freemarker.template.TemplateTransformModel;
    public class UpperCaseTransform implements TemplateTransformModel {
        public Writer getWriter(Writer out, Map args) {
            return new UpperCaseWriter(out);
        }
        private class UpperCaseWriter extends Writer {
           
            private Writer out;
               
            UpperCaseWriter (Writer out) {
                this.out = out;
            }
            public void write(char[] cbuf, int off, int len)
                    throws IOException {
                out.write(new String(cbuf, off, len).toUpperCase());
            }
            public void flush() throws IOException {
                out.flush();
            }
            public void close() {
            }
        }
    }

    然后將此對象put到數據模型中, 如下所示:

    root.put("upcase", new UpperCaseTransform());
    

    view(ftl)頁面中可以如下方式使用:

    <@upcase>
    hello world
    

    打印輸出:

    HELLO WORLD
    寫在最后

    歡迎關注喜歡、和點贊后續將推出更多的工具集教程,敬請期待。
    歡迎關注我的微信公眾號獲取更多更全的學習資源,視頻資料,技術干貨!

    公眾號回復“學習”,拉你進程序員技術討論群干貨資源第一時間分享。

    公眾號回復“視頻”,領取800GJava視頻學習資源。

    公眾號回復“全棧”,領取1T前端Java產品經理微信小程序Python等資源合集大放送。





    公眾號回復“慕課”,領取1T慕課實戰學習資源。








    公眾號回復“實戰”,領取750G項目實戰學習資源。

    公眾號回復“面試”,領取8G面試實戰學習資源。



    文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

    轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/77589.html

    相關文章

    • 具集核心教程 | 第三篇: Thymeleaf模板引擎入門進階

      摘要:介紹簡單說,是一個跟類似的模板引擎,它可以完全替代。不包含標記刪除但刪除其所有的孩子。公眾號回復全棧,領取前端,,產品經理,微信小程序,等資源合集大放送。公眾號回復面試,領取面試實戰學習資源。 thymeleaf介紹 簡單說, Thymeleaf 是一個跟 Velocity、FreeMarker 類似的模板引擎,它可以完全替代 JSP。相較與其他的模板引擎,它有如下三個極吸引人的特點:...

      abson 評論0 收藏0
    • 具集核心教程 | 第四篇: Velocity模板引擎入門進階

      摘要:是一個基于的模板引擎。模板中未被定義的變量將被認為是一個字符串。公眾號回復全棧,領取前端,,產品經理,微信小程序,等資源合集大放送。公眾號回復面試,領取面試實戰學習資源。 Velocity是一個基于java的模板引擎(template engine)。它允許任何人僅僅簡單的使用模板語言(template language)來引用由java代碼定義的對象。 當Velocity應用于web...

      leon 評論0 收藏0
    • 筆記 - 收藏集 - 掘金

      摘要:目錄如何用提高效率后端掘金經常有人說我應該學一門語言,比如之類,但是卻不知道如何入門。本文將通過我是如何開發公司年會抽獎系統的后端掘金需求出現年會將近,而年會抽獎環節必不可少,但是抽獎系統卻還沒有。 云盤一個個倒下怎么辦?無需編碼,手把手教你搭建至尊私享云盤 - 工具資源 - 掘金微盤掛了,360倒了,百度云盤也立了Flag。能讓我們在云端儲存分享文件的服務越來越少了。 買一堆移動硬盤...

      Alex 評論0 收藏0

    發表評論

    0條評論

    趙連江

    |高級講師

    TA的文章

    閱讀更多
    最新活動
    閱讀需要支付1元查看
    <