转载。。。

macro, nested, return
语法

<#macro name param1 param2 ... paramN>
...
<#nested loopvar1, loopvar2, ..., loopvarN>
...
<#return>
...
</#macro>
用例
<#macro test foo bar="Bar" baaz=-1>
Test text, and the params: ${foo}, ${bar}, ${baaz}
</#macro>
<@test foo="a" bar="b" baaz=5*5-2/>
<@test foo="a" bar="b"/>
<@test foo="a" baaz=5*5-2/>
<@test foo="a"/>

输出

Test text, and the params: a, b, 23
Test text, and the params: a, b, -1
Test text, and the params: a, Bar, 23
Test text, and the params: a, Bar, -1

定义循环输出的宏

<#macro list title items>
<p>${title?cap_first}:
<ul>
       <#list items as x>
         <li>${x?cap_first}
       </#list>
</ul>
</#macro>
<@list items=["mouse", "elephant", "python"] title="Animals"/>

输出结果
<p>Animals:
<ul>
         <li>Mouse
         <li>Elephant
         <li>Python
</ul>

包含body的宏

<#macro repeat count>
<#list 1..count as x>
       <#nested x, x/2, x==count>
</#list>
</#macro>
<@repeat count=4 ; c halfc last>
${c}. ${halfc}<#if last> Last!</#if>
</@repeat>

输出

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

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

注意在使用的时候:别忘了双引号。

<#import "/pagelibs/book.ftl" as book>

<@workorder.price value="${book.price}" />

宏Macro
宏是在模板中使用macro指令定义

l.1 基本用法

宏是和某个变量关联的模板片断,以便在模板中通过用户定义指令使用该变量,下面是一个例子:

<#macro greet> <font size="+2">Hello Joe!</font></#macro>
调用宏时,与使用FreeMarker的其他指令类似,只是使用@替代FTL标记中的#。

<@greet></@greet> <#--<@greet/>-->
在macro指令中可以在宏变量之后定义参数,如:

<#macro greet person> <font size="+2">Hello ${person}!</font></#macro>
可以这样使用这个宏变量:

<@greet person="Fred"/>
但是下面的代码具有不同的意思:

<@greet person=Fred/>
这意味着将Fred变量的值传给person参数,该值不仅是字符串,还可以是其它类型,甚至是复杂的表达式。

宏可以有多参数,下面是一个例子:

<#macro greet person color> <font size="+2" color="${color}">Hello ${person}!</font></#macro>
可以这样使用该宏变量,其中参数的次序是无关的:

<@greet person="Fred" color="black"/>
可以在定义参数时指定缺省值,否则,在调用宏的时候,必须对所有参数赋值:

<#macro greet person color="black"> <font size="+2" color="${color}">Hello ${person}!</font></#macro>
注意:宏的参数是局部变量,只能在宏定义中有效。

嵌套内容
FreeMarker的宏可以有嵌套内容,<#nested>指令会执行宏调用指令开始和结束标记之间的模板片断,举一个简单的例子:

<#macro border> <table border=4 cellspacing=0 cellpadding=4><tr><td>    <#nested> </td></tr></table></#macro>
执行宏调用:

<@border>The bordered text</@border>

输出结果:

<table border=4 cellspacing=0 cellpadding=4><tr><td>    The bordered text </td></tr></table>

<#nested>指令可以被多次调用,每次都会执行相同的内容。

<#macro do_thrice> <#nested> <#nested> <#nested></#macro><@do_thrice> Anything.</@do_thrice>

FMPP 输出结果:

Anything.Anything.Anything.

嵌套内容可以是有效的FTL,下面是一个有些复杂的例子,我们将上面三个宏组合起来:

<@border> <ul> <@do_thrice>    <li><@greet person="Joe"/> </@do_thrice> </ul></@border>

输出结果:

<table border=4 cellspacing=0 cellpadding=4><tr><td> <ul>    <li><font size="+2">Hello Joe!</font><li><font size="+2">Hello Joe!</font><li><font size="+2">Hello Joe!</font></ul></tr></td></table>

宏定义中的局部变量对嵌套内容是不可见的,例如:

<#macro repeat count> <#local y = "test"> <#list 1..count as x>    ${y} ${count}/${x}: <#nested> </#list></#macro><@repeat count=3>${y?default("?")} ${x?default("?")} ${count?default("?")}</@repeat>

输出结果:

test 3/1: ? ? ?test 3/2: ? ? ?test 3/3: ? ? ?

在宏定义中使用循环变量
nestted指令也可以有循环变量(循环变量的含义见下节),调用宏的时候在宏指令的参数后面依次列出循环变量的名字,格式如下:

<@ macro_name paramter list; loop variable list[,]>

例如:

<#macro repeat count> <#list 1..count as x>    <#nested x, x/2, x==count> </#list></#macro><@repeat count=4 ; c, halfc, last> ${c}. ${halfc}<#if last> Last!</#if></@repeat>

这里count是宏的参数,c, halfc,last则为循环变量,输出结果:

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

循环变量和宏标记指定的不同不会有问题,如果调用时少指定了循环变量,那么多余的值不可见。调用时多指定了循环变量,多余的循环变量不会被创建:

<@repeat count=4 ; c, halfc, last> ${c}. ${halfc}<#if last> Last!</#if></@repeat><@repeat count=4 ; c, halfc> ${c}. ${halfc}</@repeat><@repeat count=4> Just repeat it...</@repeat>
在模板中定义变量
在模板中定义的变量有三种类型:

plain变量:可以在模板的任何地方访问,包括使用include指令插入的模板,使用assign指令创建和替换。 
局部变量:在宏定义体中有效,使用local指令创建和替换。 
循环变量:只能存在于指令的嵌套内容,由指令(如list)自动创建;宏的参数是局部变量,而不是循环变量 
局部变量隐藏(而不是覆盖)同名的plain变量;循环变量隐藏同名的局部变量和plain变量,下面是一个例子:

<#assign x = "plain">${x} <#-- we see the plain var. here --><@test/>6. ${x} <#-- the value of plain var. was not changed --><#list ["loop"] as x>    7. ${x} <#-- now the loop var. hides the plain var. -->    <#assign x = "plain2"> <#-- replace the plain var, hiding does not mater here -->    8. ${x} <#-- it still hides the plain var. --></#list>9. ${x} <#-- the new value of plain var. --><#macro test> 2. ${x} <#-- we still see the plain var. here --> <#local x = "local"> 3. ${x} <#-- now the local var. hides it --> <#list ["loop"] as x>    4. ${x} <#-- now the loop var. hides the local var. --> </#list> 5. ${x} <#-- now we see the local var. again --></#macro>
输出结果:

1. plain 2. plain 3. local 4. loop 5. local 6. plain 7. loop 8. loop 9. plain2
内部循环变量隐藏同名的外部循环变量,如:

<#list ["loop 1"] as x> ${x} <#list ["loop 2"] as x>    ${x}    <#list ["loop 3"] as x>      ${x}    </#list>    ${x} </#list> ${x}</#list>
输出结果:

loop 1    loop 2      loop 3    loop 2 loop 1
模板中的变量会隐藏(而不是覆盖)数据模型中同名变量,如果需要访问数据模型中的同名变量,使用特殊变量global,下面的例子假设数据模型中的user的值是Big Joe:

<#assign user = "Joe Hider">${user}          <#-- prints: Joe Hider -->${.globals.user} <#-- prints: Big Joe -->
名字空间
通常情况,只使用一个名字空间,称为主名字空间,但为了创建可重用的宏、变换器或其它变量的集合(通常称库),必须使用多名字空间,其目的是防止同名冲突

创建库
下面是一个创建库的例子(假设保存在lib/my_test.ftl中):

<#macro copyright date> <p>Copyright (C) ${date} Julia Smith. All rights reserved. <br>Email: ${mail}</p></#macro> <#assign mail = "jsmith@acme.com">
使用import指令导入库到模板中,Freemarker会为导入的库创建新的名字空间,并可以通过import指令中指定的散列变量访问库中的变量:

<#import "/lib/my_test.ftl" as my><#assign mail="fred@acme.com"><@my.copyright date="1999-2002"/>${my.mail}${mail}
输出结果:

<p>Copyright (C) 1999-2002 Julia Smith. All rights reserved. <br>Email: jsmith@acme.com</p>jsmith@acme.comfred@acme.com
可以看到例子中使用的两个同名变量并没有冲突,因为它们位于不同的名字空间。还可以使用assign指令在导入的名字空间中创建或替代变量,下面是一个例子:

<#import "/lib/my_test.ftl" as my>${my.mail}<#assign mail="jsmith@other.com" in my>${my.mail}
输出结果:

jsmith@acme.comjsmith@other.com
数据模型中的变量任何地方都可见,也包括不同的名字空间,下面是修改的库:

<#macro copyright date> <p>Copyright (C) ${date} ${user}. All rights reserved.</p></#macro><#assign mail = "${user}@acme.com">
假设数据模型中的user变量的值是Fred,则下面的代码:

<#import "/lib/my_test.ftl" as my><@my.copyright date="1999-2002"/>${my.mail}
输出结果:

<p>Copyright (C) 1999-2002 Fred. All rights reserved.</p>Fred@acme.com
选择Velocity还是FreeMarker?
Velocity是另外一个优秀的模板引擎,它是FreeMarker的有力竞争者。但是它的模板语言不够强大,也不够严谨。

除开FreeMarker内建的Velocity所没有的强大函数外,FreeMarker还在以下方面更胜一筹:

与Velocity 相比,FreeMarker 对表现逻辑和业务逻辑的划分更为严格,Freemarker在模板中不允许对Servlet API进行直接操作(而Velocity可以),如FreeMarker 中禁止对HttpServletRequest 对象直接访问(但可以访问HttpServletRequest对象中的Attribute)。通过更加严格的隔离机制,牵涉逻辑处理的操作被强制转移到 逻辑层。从而完全保证了层次之间的清晰性。

另外一个Velocity无法实现的特性,也是最具备实际意义的特性:FreeMarker提供了强大的宏。

此外,FreeMarker对JSP Tag提供了良好支持。我们可以将FreeMarker看作是仅允许使用TAG的JSP页面(实际上,FreeMarker的表达式语法与EL语法也非常 类似)。从开发角度而言,只允许使用TAG的JSP页面,已经在很大程度上保证了页面表现逻辑与业务逻辑的分离。程序员在JSP Script中混杂逻辑代码的原因,大部分是出于慵懒,只要无法在页面模板中直接编写Java代码,相信程序员也不会去专门编写一个JSP TAG来刻意违反层次划分原则。

 
如果 是Boolean类型  使用 ${bool?c} 来显示---
 
 

freemarker macro 使用的更多相关文章

  1. 转 freemarker macro(宏)的使用

    有人说用freemarker,但没有用到它的宏(macro),就=没有真正用过freemarker.说的就是宏是freemarker的一大特色. 宏的定义可以查看相关的文档,里面介绍得很清楚,下面来看 ...

  2. FreeMarker教程

    一.什么是模板引擎,为什么要用模板引擎 在B/S程式设计中,常常有美工和程序员二个角色,他们具有不同专业技能:美工专注于表现——创建页面.风格.布局.效果等等可视元素:而程序员则忙于创建程式的商业流程 ...

  3. Freemarker 的 Shiro 标签使用详解

    一.引入依赖(已解决版本冲突) <!-- shiro-freemarker-tags start --> <dependency> <groupId>net.min ...

  4. SpringMVC 集成 Freemarker 模板引擎

    本文通过 maven 项目中集成 1.引入 SpringMVC 与 Freemarker 需要的依赖 <!-- SpringMVC --> <dependency> <g ...

  5. Shiro权限项目

    目录 环境配置 spring容器 springmvc freemarker mybatis shiro 工具类 TokenManager.java Result.java 功能实现 登录 注册 个人中 ...

  6. FreeMarker学习(宏<#macro>的使用)

    原文链接:https://my.oschina.net/weiweiblog/blog/506301?p=1 用户定义指令-使用@符合来调用  有两种不同的类型:Macro(宏)和transform( ...

  7. freemarker.template.TemplateException:Macro has no such argument:params

    1.错误描述 freemarker.template.TemplateException:Macro mainSelect has no such argument:params 2.错误原因 在宏定 ...

  8. freemarker.template.TemplateException:Error executing macro:mainSelect

    1.错误描述 freemarker.template.TemplateException:Error executing macro:mainSelect require parameter:id i ...

  9. 异常-----freemarker.template.TemplateException:Error executing macro:mainSelect

    1.错误描述 freemarker.template.TemplateException:Error executing macro:mainSelect require parameter:id i ...

随机推荐

  1. Firemonkey Android IOS 图标

    图标很多

  2. Excel向数据库插入数据和数据库向Excel导出数据

    为了熟悉java里工作簿的相关知识点,所以找了“Excel向数据库插入数据和数据库向Excel导出数据”的功能来实现. 注意事项:1,mysql数据库: 2,需要导入的jar包有 jxl.jar,my ...

  3. EF CODEFIRST WITH ORACLE 存储过程

    EF  CODEFIRST WITH ORACLE 解决存储过程一直没找到解决方案 所以最后也没办法还是用了最基本的解决方案 采用Oracle.ManagedDataAccess提供的ADO基础访问类 ...

  4. mysql:mysql Access denied for user root@

    最近用本地Navicat连接集群的mysql,报了上述的错误,我认为是权限问题 之前试过赋权限给所有人,但是我这边还是连接不上,无奈,试试只分给我一个IP 开始:mysql -uroot -p //先 ...

  5. 【摘自张宴的"实战:Nginx"】nginx配置

    user nobody;worker_processes 2; #error_log logs/error.log;error_log logs/error.log notice;#error_log ...

  6. SPOJ LCMSUM - LCM Sum

    题意是求: $\sum_{i = 1}^{n}lcm(i, n)$ $= \sum_{i = 1}^{n}\frac{ni}{gcd(i, n)}$ $= n\sum_{i = 1}^{n}\frac ...

  7. python3-字典的循环

    # Auther: Aaron Fan info = { 'stu1102': 'LongZe Luola', 'stu1103': 'XiaoZe Maliya', 'stu1106': 'Alex ...

  8. php 函数追踪扩展 phptrace

    php 函数追踪扩展 phptrace 介绍 phptrace 是一个低开销的用于跟踪.分析 php 运行情况的工具. 它可以跟踪 php 在运行时的函数调用.请求信息.执行流程.并且提供有过滤器.统 ...

  9. 巧用 git rebase 将某一部分 commit 复制到另一个分支

    一.为什么需要将一个 commit 复制到其他分支上去呢? 在我们的实际开发的过程中,我们的项目中会存在多个分支. 在某些情况下,可能需要将某一个分支上的 commit 复制到另一个分支上去.   二 ...

  10. 《Head First Servlets & JSP》-10-定制标记开发

    标记文件:很想include,但是比include更好 建立和使用标记文件的最简方法 取一个被包含文件(如Header.jsp),把它重命名为带有一个.tag扩展名(Header.tag): 把标记文 ...