FreeMarker中的自定义指令和四种变量

完整版见https://jadyer.github.io/




FreeMarker最强大之处莫过于它的自定义指令了


首先是我们自定义的工具类FreeMarkerUtil.java

package com.jadyer.util;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.Writer;
import java.util.Map;

import freemarker.template.Configuration;
import freemarker.template.Template;
import freemarker.template.TemplateException;

public class FreeMarkerUtil {
	/**
	 * 获取指定目录下的模板文件
	 * @param name       模板文件的名称
	 * @param pathPrefix 模板文件的目录
	 */
	public Template getTemplate(String name, String pathPrefix) throws IOException{
		Configuration cfg = new Configuration(); //通过FreeMarker的Configuration对象可以读取ftl文件
		cfg.setClassForTemplateLoading(this.getClass(), pathPrefix); //设置模板文件的目录
		cfg.setDefaultEncoding("UTF-8");       //Set the default charset of the template files
		Template temp = cfg.getTemplate(name); //在模板文件目录中寻找名为"name"的模板文件
		return temp; //此时FreeMarker就会到类路径下的"pathPrefix"文件夹中寻找名为"name"的模板文件
	}
	
	/**
	 * 根据模板文件输出内容到控制台
	 * @param name       模板文件的名称
	 * @param pathPrefix 模板文件的目录
	 * @param rootMap    模板的数据模型
	 */
	public void print(String name, String pathPrefix, Map<String,Object> rootMap) throws TemplateException, IOException{
		this.getTemplate(name, pathPrefix).process(rootMap, new PrintWriter(System.out));
	}
	
	/**
	 * 根据模板文件输出内容到指定的文件中
	 * @param name       模板文件的名称
	 * @param pathPrefix 模板文件的目录
	 * @param rootMap    模板的数据模型
	 * @param file       内容的输出文件
	 */
	public void printFile(String name, String pathPrefix, Map<String,Object> rootMap, File file) throws TemplateException, IOException{
		Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file), "UTF-8"));
		this.getTemplate(name, pathPrefix).process(rootMap, out); //将模板文件内容以UTF-8编码输出到相应的流中
		if(null != out){
			out.close();
		}
	}
}

下面是位于//src//ftl//包中用于演示自定义指令的用法的macro.ftl

<#-- 自定义指令 -->

<#-- 自定义一个名字为myMacro的指令 -->
<#-- 该指令的作用就是输出<h2>Hi:macro</h2>,调用该指令时使用<@myMacro/>即可 -->
<#macro myMacro>
	<h1>Hi:macro</h1>
</#macro>
<@myMacro/>


<#-- 自定义一个带有参数的名字为myMacro22的指令 -->
<#-- 这里myMacro22后面跟的都是参数,这里有两个参数,分别为num和name -->
<#macro myMacro22 num name>
	<#list 1..num as n>
		<h2>hello:${name}${n}${n}</h2>
	</#list>
</#macro>
<@myMacro22 4 "blackjade"/>
<@myMacro22 num=4 name="Jadyer"/>


<#-- 为指令参数定义初始值 -->
<#-- 此时就可以直接调用<@myMacro33/>而不会报错了。注意:指定初始值的参数要放在未指定初始值参数的后面 -->
<#macro myMacro33 num=3 name="JADYER">
	<#list 1..num as n>
		<h3>welcome:${name}${n}${n}</h3>
	</#list>
</#macro>
<@myMacro33/>
<@myMacro33 num=4 name="myblog"/>


<#-- 使用<#nested/>可以输出指令中的内容,输出的次数与<#nested/>的数量有关 -->
<#macro myMacro44>
	<#nested/>
	<#nested/>
</#macro>
<@myMacro44>
	<h4>http://blog.csdn.net/jadyer</h4>
</@myMacro44>


<#-- 实际输出时会根据<#nested/>的数量来打印,而且<#nested/>还可以传参给<@myMacro55> -->
<#macro myMacro55>
	<#nested 55 66/>
	<#nested 55 77/>
	<#nested 55 88/>
</#macro>
<@myMacro55;x,y>
	<h5>It is ${x}and${y}</h5>
</@myMacro55>

下面是位于//src//ftl//包中用于演示四种变量的用法的assign.ftl

<#--
变量,分为以下四类
模型变量----rootMap中的变量
模板变量----使用<#assign>定义的变量
局部变量----指令中的变量
循环变量----循环中的变量
-->

<#assign myname="Jadyer"/>
<#-- 此时模板变量名称与模型变量名称相同,虽然打印的是模板变量值,但模型变量值并没有被覆盖,而被隐藏了 -->
<#-- 并且这两个变量是存在于不同空间中的,如此才给人一种"被隐藏"的意识,这是FreeMarker中很重要的一个概念 -->
<#-- 在调用变量的时候,实际上它会先到模板变量中查找,没有则去模型变量中查找 -->
${myname}
<#-- 通过这种方式就可以访问模型变量了 -->
${.globals.myname}


<#-- 调用该指令后,会将模板变量myname的值覆盖为JADYER。注意此时是覆盖,不是隐藏,相当于模板的覆盖模板的 -->
<#macro myMacro>
	<#assign myname="JADYER"/>
</#macro>
<#-- 由于此时还没有调用myMacro指令,故打印的仍是最上面定义的模板变量值Jadyer -->
${myname}
<@myMacro/>
<#-- 由于此时已经调用了myMacro指令,故打印的是模板中的变量被覆盖后的值JADYER -->
${myname}


<#-- 上面那种在指令中定义变量的方式存在风险,容易误将模板变量中同名的变量覆盖 -->
<#-- 而使用<#local/>定义的变量则不会覆盖模板变量,且此时的变量就属于局部变量了 -->
<#macro myMacro22>
	<#local myname="http://blog.csdn.net/jadyer"/>
	${myname}
</#macro>
${myname}
<@myMacro22/>
${myname}


<#-- 循环变量的作用域为循环体内 -->
<#list 1..3 as myname>
	${myname}
</#list>
${myname}

最后是使用JUnit4.x编写的测试类FreeMarkerTest.java

package com.jadyer.test;

import java.io.IOException;
import java.util.HashMap;
import java.util.Map;

import org.junit.Before;
import org.junit.Test;

import com.jadyer.util.FreeMarkerUtil;

import freemarker.template.TemplateException;

public class FreeMarkerTest {
	String pathPrefix;
	Map<String,Object> rootMap;
	FreeMarkerUtil fmu;
	
	@Before
	public void setUp(){
		pathPrefix = "/ftl";
		rootMap = new HashMap<String,Object>();
		fmu = new FreeMarkerUtil();
	}
	
	@Test
	public void printMacro() throws TemplateException, IOException{
		fmu.print("macro.ftl", pathPrefix, rootMap);
	}
	
	@Test
	public void printAssign() throws TemplateException, IOException{
		rootMap.put("myname", "XuanYu"); //这个变量就是数据模型中的变量,即模型变量
		fmu.print("assign.ftl", pathPrefix, rootMap);
	}
}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值