FreeMarker 学习笔记

FreeMarker语言

         FreeMarker 是一个非常优秀的模板引擎,这个模板引擎可用于任何场景,FreeMarker负责将数据模型中的数据合并到模板中,从而生成标准输出.界面开发人员只需 要关于界面(也就是模板文件)的开发,而业务逻辑开发者只需要负责将需要显示的数据填入数据模型

     FreeMarker负责合并数据模型和模板, 从而生成标准输出.FreeMarker特别适应于MVC模式的Web应用,虽然FreeMarker具有一些编程能力,但这种编程能力非常有限,无法实现业务逻 辑, 只能提供一些数据格式的转换功能.因此,通常由Java程序准备要显示的数据,由FreeMarker模板引擎来生成页面,而FreeMarker模板则 提供页面布局支持.(好处:严格实现MVC分离)


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

FreeMarker语言概述

 

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

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

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

FreeMarker是免费的。

FreeMarker特性

通用目标

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

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

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

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

强大的模板语言

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

在模板中创建和改变变量

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

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

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

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

通用数据模型

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

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

为Web准备

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

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

支持JSP标记库

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

第一个FreeMarker程序

1.       建立一个普通的java项目:testFreeMarker

2.       引入freemarker.jar

3.       在项目目录下建立模板目录:templates

4.       在templates目录下,建立a.ftl模板文件,内容如下:

你好啊,${user},今天你的精神不错!

5.  建立com.sxt.test.freemarker包,然后建立Test1.java文件,内容如下:

package com.sxt.test.freemarker;

 

import java.io.File;

import java.io.OutputStreamWriter;

import java.io.Writer;

import java.util.HashMap;

import java.util.Map;

 

import freemarker.template.Configuration;

import freemarker.template.DefaultObjectWrapper;

import freemarker.template.Template;

 

public class Test1 {

         public static void main(String[] args) throws Exception {

                   //创建Freemarker配置实例

                   Configuration cfg = new Configuration();

                  

                   cfg.setDirectoryForTemplateLoading(new File("templates"));

                  

                   //创建数据模型

                   Map root = new HashMap();

                   root.put("user", "老高");

                  

                   //加载模板文件

                   Template t1 = cfg.getTemplate("a.ftl");

                  

                  //显示生成的数据,//将合并后的数据打印到控制台

      Writer out = new OutputStreamWriter(System.out);

      t1.process(root, out);

      out.flush();

 

      //显示生成的数据,//将合并后的数据直接返回成字符串!

//    StringWriter out = new StringWriter();  

//    t1.process(root, out);

//    out.flush();

//    String temp = out.toString();

//    System.out.println(temp);        }

}

6.      编译和运行Test1.java文件,控制台打印:

数据类型

一、直接指定值

直接指定值可以是字符串、数值、布尔值、集合及Map对象。


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


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

插值的打印都是给用户看的,而不是给计算机的。有时候这样并不好,比如你要打印数据库记录的主键,用来作为URL中的一部分或HTML表单的隐藏域来作为提交内容,或者要打印CSS/JavaScript的数字。这些值都是给计算机程序去识别的而不是用户,很多程序对数字格式的要求非常严格,它们只能理解一部分简单的美式数字格式。那样的话,可以使用内建函数c(代表计算机)来解决这个问题,比如:<a href="/shop/productdetails?id=${product.id?c}">Details...</a>

当数据超过3位的时候,freemarker会自动用逗号截取 格式如:2,008

如何解决呢?

1> 加.toString(),如:${(userId).toString()}
2> 加?c,如:
${(userId)?c}
3> freemarker配置文件freemarker.properties加上number_format=#


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

<#-- 布尔值输出 -->
 <#assign flag = true />
 ${flag?string}<br><#--输出结果:true-->
 ${flag?string("1","0")}><#--输出结果:1-->


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

<#assign nums=[1,2,3,4,5] />
 <#list nums as num>
  ${num}
 </#list><#--输出结果:12345 -->

<#list 1..5 as num>

${num}

</#list><#--输出结果:12345 -->
 <#assign nums1=1..10 />
 <#list nums1 as num>
  ${num}
 </#list><#--输出结果:12345678910  -->
 <#assign nums2=nums1[0..4]/>
 <#list nums2 as num>
  ${num}
 </#list><#--输出结果:12345 -->

<#assign name="张三">
<#-- 序列 拆分-->
 ${"helloworld!"[0..4]},${name}<br> <#-- 输出结果:hello,张三  --->

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

 <#-- 遍历map,map的key值必须为字符串形式  -->
 <#assign maps={"1":"张三","2":"李四"} />
 ${maps["1"]} <#-- 输出结果:张三 --->

 <#assign keys=maps?keys>
 <#list keys as key>
  ${key}:${maps[key]}
 </#list><#- 输出结果:1:张三2:李四--->

 <#listmaps?keys as key>
  ${key}:${maps[key]}
 </#list><#- 输出结果:1:张三2:李四--->

 <#list maps.keySet() as key>
  ${key}:${maps[key]}
 </#list><#- 输出结果:1:张三2:李四-->

 <#assign user={"name":"张三","password":"1234"} />
 ${user.name} : ${user.password} <#--或者${user["password"]}-->
<#- 输出结果:张三:1234--->

6.时间对象

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

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

 <#-- string类型日期转换为日期 date日期  datetime日期和时间 time时间-->
 ${"1949-10-01 00:00:00"?date("yyyy-MM-dd")}<br> <#- 输出结果:1949-10-1--->

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! -->

${user.group.name!} //freemarker只会判断group.name是否是空值, 可以使用${(user.group.name)!“admin”}

$ {(a,b)!"没有a,b元素"}

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

        

数据类型常见示例

 直接指定值

  字符串: "Foo"或者'Foo'"It's \"quoted\""r"C:\raw\string"

  数字:123.45

  布尔值:true, false

  序列:["foo", "bar", 123.45], 1..100

  哈希表:{"name":"green mouse", "price":150}

  检索变量   顶层变量:user

  从哈希表中检索数据:user.name, user[“name”]

  从序列中检索:products[5]

  特殊变量:.main

  字符串操作

  插值(或连接):"Hello ${user}!"(或"Free" + "Marker"

  获取一个字符:name[0]

  序列操作

  连接:users + ["guest"]

  序列切分:products[10..19]   products[5..]

  哈希表操作

  连接:passwords + {"joe":"secret42"}

  算数运算: (x * 1.5 + 10) / 2 - y % 100

  比较运算: x == y,   x != y,   x < y,   x > y,   x >= y,   x <= y,

x &lt; y, 等等

   逻辑操作:!registered && (firstVisit || fromEurope)

   内建函数:name?upper_case

   方法调用:repeat("What", 3)

  处理不存在的值

  默认值:name!"unknown" 或者(user.name)!"unknown" 或者

name! 或者  (user.name)!

  检测不存在的值:name??或者(user.name)??

        参考:运算符的优先级

模板开发语句

if指令

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

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

if语句测试:

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

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

if else 语句测试:

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

    及格!

<#else>

    不及格!

</#if>

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

if else if else语句测试:

<#if randomgte 90>

    优秀!

<#elseif randomgte 80>

    良好!

<#else>

    一般! 

</#if>

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

 

list指令

      List list = new ArrayList();

      list.add(new Address("中国","北京"));

      list.add(new Address("中国","上海"));

      list.add(new Address("美国","纽约"));

      root.put("lst", list);

测试list指令:

<#list lst as dizhi >

    <b>${dizhi.country}</b> <br/>

</#list>

思考问题:<c:forEach> status属性。在此处如何实现?

 

控制台打印:

测试list语句:

   <b>中国</b> <br/>

   <b>中国</b> <br/>

   <b>美国</b> <br/>

 

include指令

增加被包含文件,放于templates目录

 

文件内容如下:

模板文件中代码如下:

测试include指令:

<#include "included.txt" />

注:使用include可能会出现覆盖的问题,可以使用import的命名空间来完成

import指令

mylib.ftl

<#assign name="hello" />

<#marcro copyright date>     

${date}

</#marcro>

使用import指令引入mylib.ftl

<#import "/libs/mylib.ftl" as my> <#--  此处my是命名空间 -->

${my.name} <#-- 输出结果:hello -->

<@my.copyright date="1999-12-31"/><#-- 输出结果:1999-12-31 -->

 

自定义指令(macro指令)

<#macro m1>   <#--定义指令m1 -->

    <b>aaabbbccc</b>

    <b>dddeeefff</b>

</#macro>

<@m1 /><@m1 />  <#--调用上面的宏指令 -->

 定义带参的宏指令:

<#macro m2 a b c >

    ${a}--${b}--${c}

</#macro>

<@m2 a="老高" b="老张" c="老马" />

nested指令:

<#macro border>

  <table border=4cellspacing=0cellpadding=4><tr><td>

   <#nested> <#-- 输出的是<@border>和</@border>之间的内容 -->

  </td></tr></table>

</#macro>

<@border >表格中的内容!/@border

 

命名空间

当运行 FTL模板时,就会有使用 assign macro指令创建的变量的集合(可能是空的),可以从前一章节来看如何使用它们。像这样的变量集合被称为 namespace命名空间。在简单的情况下可以只使用一个命名空间,称之为 main namespace主命名空间。因为通常只使用本页上的命名空间,所以就没有意识到这点。

   如果想创建可以重复使用的宏,函数和其他变量的集合,通常用术语来说就是引用library库。使用多个命名空间是必然的。只要考虑你在一些项目中,或者想和他人共享使用的时候,你是否有一个很大的宏的集合。但要确保库中没有宏(或其他变量)名和数据模型中变量同名,而且也不能和模板中引用其他库中的变量同名。通常来说,变量因为名称冲突也会相互冲突。所以要为每个库中的变量使用不同的命名空间。

定义b.ftl文件:

<#macro copyright date>

  <p>Copyright (C) ${date} 北京尚学堂.</p>

</#macro>

<#assign mail = "bjsxt@163.com">

a.ftl文件中引入b.ftl,从而可以使用b.ftl中定义的宏和变量:

测试命名空间:

<#import "b.ftl" asbb  />

<@bb.copyright date="2010-2011" />

${bb.mail}

<#assign mail="my@163.com"  />

${mail}

<#assign mail="my@163.com" inbb  />

${bb.mail}

执行后,控制台打印:

测试命名空间:

 <p>Copyright (C) 2010-2011 北京尚学堂.</p>

bjsxt@163.com

my@163.com

my@163.com

 

命名空间命名规则

如果你为 Example公司工作,它们拥有 www.example.com 网的主页,你的工作是开发

一个部件库,那么要引入你所写的 FTL的路径应该是:

/lib/example.com/widget.ftl

注意到 www已经被省略了。第三次路径分割后的部分可以包含子目录,可以像下面这

样写:  /lib/example.com/commons/string.ftl

一个重要的规则就是路径不应该包含大写字母,为了分隔词语,使用下划线_,就像

wml_form(而不是 wmlForm)。

如果你的工作不是为公司或组织开发库,也要注意,你应该使用项目主页的 URL,比如

/lib/example.sourceforge.net/example.ftl/lib/geocities.com/jsmith/example.ftl

Servlet中使用Freemarker

参考Freemarker包中example目录下webapp1项目!

struts2中整合FreeMarker

1.解压struts2-core-X.X.X.jar文件,把在META-INF文件夹下面的struts-tags.tld文件复制到WEB-INF文件夹下。   freemarkjar导入到工程中

.web.xml文件中配置freemark同时启动JSPSupportServlet.代码如下:

<servlet>
        <servlet-name>freemarker</servlet-name>
        <servlet-class>
            freemarker.ext.servlet.FreemarkerServlet
        </servlet-class>

        <!--下面的配置freemarke的ftl文件的位置 -->
        <init-param>
            <param-name>TemplatePath</param-name>
            <param-value>/</param-value>
        </init-param>
        <!-- 是否和服务器(tommcat)一起启动。-->
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>freemarker</servlet-name>
        <url-pattern>*.ftl</url-pattern>
    </servlet-mapping>


<servlet>
  <!-- define a JspSupportServlet Object -->
  <servlet-name>JspSupportServlet</servlet-name>
  <servlet-class>org.apache.struts2.views.JspSupportServlet</servlet-class>
  <!-- setting JspSupportServlet auto start -->
  <load-on-startup>1</load-on-startup>
 </servlet>

 

3.在FreeMarker模板中使用assign指令导入标签库。代码如下

<#assign s=JspTaglibs["/WEB-INF/struts-tags.tld"] /> 注:这里我把struts-tags.tld放在WEB-INF下面,

4.现在我们可以在FreeMarker模板中使用标签了

1.      使用list标签遍历list容器时,如何获取索引(下划线加index即可获取)

 

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

测试list索引_index

<#list list as city>

         ${city}<br/>${city_index}

         <#if city_index==1>

                   <#break>

         </#if>

</#list>

**************************************

测试list中国_has_next

<#list list as city>

         ${city}<br/>${city_index}

         <#if city_has_next>

                  我有下一项!------- ${city_index}

         </#if>

</#list>

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FreeMarker的模板文件并不比HTML页面复杂多少,FreeMarker模板文件主要由如下4个部分组成: 1,文本:直接输出的部分 2,注释:<#-- ... -->格式部分,不会输出 3,插值:即${...}或#{...}格式的部分,将使用数据模型中的部分替代输出 4,FTL指令:FreeMarker指定,和HTML标记类似,名字前加#予以区分,不会输出 下面是一个FreeMarker模板的例子,包含了以上所说的4个部分 <html><br> <head><br> <title>Welcome!</title><br> </head><br> <body><br> <#-- 注释部分 --><br> <#-- 下面使用插值 --> <h1>Welcome ${user} !</h1><br> <p>We have these animals:<br> <u1><br> <#-- 使用FTL指令 --> <#list animals as being><br> <li>${being.name} for ${being.price} Euros<br> <#list><br> <u1><br> </body><br> </html> 1, FTL指令规则 在FreeMarker中,使用FTL标签来使用指令,FreeMarker有3种FTL标签,这和HTML标签是完全类似的. 1,开始标签:<#directivename parameter> 2,结束标签:</#directivename> 3,空标签:<#directivename parameter/> 实际上,使用标签时前面的符号#也可能变成@,如果该指令是一个用户指令而不是系统内建指令时,应将#符号改成@符号. 用户指令用@ 使用FTL标签时, 应该有正确的嵌套,而不是交叉使用,这和XML标签的用法完全一样.如果全用不存在的指令,FreeMarker不会使用模板输出,而是产生一个错误消息.FreeMarker会忽略FTL标签中的空白字符.值得注意的是< , /> 和指令之间不允许有空白字符. 2, 插值规则 FreeMarker的插值有如下两种类型:1,通用插值${expr};2,数字格式化插值:#{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; m1} <#-- 输出2.6 --> #{y; m1} <#-- 输出4.0 --> #{x; m1M2} <#-- 输出2.58 --> #{x; m1M2} <#-- 输出4.0 --> 3, 表达式 表达式是FreeMarker模板的核心功能,表达式放置在插值语法${}之中时,表明需要输出表达式的值;表达式语法也可与FreeMarker 标签结合,用于控制输出.实际上FreeMarker的表达式功能非常强大,它不仅支持直接指定值,输出变量值,也支持字符串格式化输出和集合访问等功能. 3.1 直接指定值 使用直接指定值语法让FreeMarker直接输出插值中的值,而不是输出变量值.直接指定值可以是字符串,数值,布尔值,集合和MAP对象. 1,字符串 直接指定字符串值使用单引号或双引号限定,如果字符串值中包含特殊字符需要转义,看下面的例子: ${"我的文件保存在C:\\盘"} ${'我名字是\"annlee\"'} 输出结果是: 我的文件保存在C:\盘 我名字是"annlee" FreeMarker支持如下转义字符: \";双引号(u0022) \';单引号(u0027) \\;反斜杠(u005C) \n;换行(u000A) \r;回车(u000D) \t;Tab(u0009) \b;退格键(u0008) \f;Form feed(u000C) \l;< \g;> \a;& \{;{ \xCode;直接通过4位的16进制数来指定Unicode码,输出该unicode码对应的字符. 如果某段文本中包含大量的特殊符号,FreeMarker提供了另一种特殊格式:可以在指定字符串内容的引号前增加r标记,在r标记后的文件将会直接输出.看如下代码: ${r"${foo}"} ${r"C:\foo\bar"} 输出结果是: ${foo} C:\foo\bar 2,数值 表达式中的数值直接输出,不需要引号.小数点使用"."分隔,不能使用分组","符号.FreeMarker目前还不支持科学计数法,所以"1E3"是错误的.在FreeMarker表达式中使用数值需要注意以下几点: 1,数值不能省略小数点前面的0,所以".5"是错误的写法 2,数值8 , +8 , 8.00都是相同的 3,布尔值 直接使用true和false,不使用引号. 4,集合 集合以方括号包括,各集合元素之间以英文逗号","分隔,看如下的例子: <#list ["星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期天"] as x> ${x} </#list> 输出结果是: 星期一 星期二 星期三 星期四 星期五 星期六 星期天 除此之外,集合元素也可以是表达式,例子如下: [2 + 2, [1, 2, 3, 4], "whatnot"] 还可以使用数字范围定义数字集合,如2..5等同于[2, 3, 4, 5],但是更有效率.注意,使用数字范围来定义集合时无需使用方括号,数字范围也支持反递增的数字范围,如5..2 5,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,属性值为value的Map实例.为了输出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"] //全部使用方括号语法 使用点语法时,变量名字有顶层变量一样的限制,但方括号语法没有该限制,因为名字可以是任意表达式的结果. Map?keys这个map当中所有的key的列表 Map?values这个map当中所有的value的列表 3.3, 字符串操作 FreeMarker的表达式对字符串操作非常灵活,可以将字符串常量和变量连接起来,也可以返回字符串的子串等. 字符串连接有两种语法: 1,使用${..}或#{..}在字符串常量部分插入表达式的值,从而完成字符串连接. 2,直接使用连接运算符+来连接字符串 例如有如下数据模型: Map root = new HashMap(); root.put("user","annlee"); 下面将user变量和常量连接起来: ${"hello, ${user}!"} //使用第一种语法来连接 ${"hello, " + user + "!"} //使用+号来连接 上面的输出字符串都是hello,annlee!,可以看出这两种语法的效果完全一样. 值得注意的是,${..}只能用于文本部分,不能用于表达式,下面的代码是错误的: <#if ${isBig}>Wow!</#if> <#if "${isBig}">Wow!</#if> 应该写成:<#if isBig>Wow!</#if> 截取子串可以根据字符串的索引来进行,截取子串时如果只指定了一个索引值,则用于取得字符串中指定索引所对应的字符;如果指定两个索引值,则返回两个索引中间的字符串子串.假如有如下数据模型: Map root = new HashMap(); root.put("book","struts2,freemarker"); 可以通过如下语法来截取子串: ${book[0]}${book[4]} //结果是su ${book[1..4]} //结果是tru
FreeMarker是一个用于生成HTML Web页面的模板引擎。它被设计用于将视图从业务逻辑中分离,通过模板来生成页面并展示数据。虽然FreeMarker具有一些编程能力,但通常由Java程序准备要显示的数据,然后由FreeMarker生成页面。它不是一个Web应用框架,而是适合作为Web应用框架的一个组件。FreeMarker与容器无关,可以应用于非Web应用程序环境。它更适合作为Model2框架(如Struts)的视图组件,也可以在模板中使用JSP标记库。FreeMarker具有强大的模板语言,支持常用的指令、循环结构、变量定义和复杂表达式等。它还提供了通用数据模型,可以方便地处理各种数据。\[2\] 在使用FreeMarker时,可以在webapp目录下创建template文件夹,并在其中创建模板文件。模板文件使用HTML标签和FreeMarker注释语法来定义页面结构和逻辑。可以通过${}语法来引用数据并在页面中展示。\[3\] 要启动使用FreeMarker的项目,需要在项目的依赖中添加FreeMarker的相关配置。可以在pom.xml文件中添加如下依赖配置: ```xml <dependency> <groupId>org.FreeMarker</groupId> <artifactId>FreeMarker</artifactId> <version>2.3.29</version> </dependency> ``` 这样就可以在项目中使用FreeMarker来生成页面了。\[1\] #### 引用[.reference_title] - *1* [FreeMarker简介及使用示例](https://blog.csdn.net/qq_39326472/article/details/131264082)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [FreeMarker详细介绍](https://blog.csdn.net/weixin_44454512/article/details/109877418)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值