首页 > 代码库 > freemarker的使用心得
freemarker的使用心得
freemarker目前虽然使用的公司不多,但是在某些时候就必须使用,当需要遍历json数据的时候,目前市面上的好多显示组件都达不到要求唯一能用的也就只有freemarker了。在java里使用的模板引擎有velocity、freemarker,至于velocity由于比较古老加之功能也不及freemarker那么强大使用的人群已经很少了。这次打算系统性的分享我学习freemarker的心得。
一 第一个freemarker程序
freemarker是一种与web容器无关的技术,应用并不局限于页面。使用java开发出来的模板引擎。因此java中使用freemarker应该算得上是一个比较好的选择了
freemarker的运行原理总结起来就是如下四点:
(1)初始化freemarker配置实例
(2)创建数据模型
(3)加载模板文件
(4)显示生成后的数据
如果要使用freemarker要做的就是把freemarker的jar包放到lib或者构建路径下面就OK了,来看下面这段程序
package org.lxh; import java.io.*; import java.util.*; import freemarker.template.Configuration; import freemarker.template.Template; public class FirstFreemarker { public static void main(String[] args) throws Exception { //创建freemarker配置实例 Configuration config=new Configuration(); config.setDirectoryForTemplateLoading(new File("template")); //创建数据模型 Map<String,String> m=new HashMap<String,String>(); m.put("name", "cry"); //加载模板文件 Template t=config.getTemplate("index.ftl"); //显示生成后的数据 Writer w=new OutputStreamWriter(System.out); t.process(m, w); w.close(); } }运行效果如下所示:
二 freemarker进阶
如果觉得freemarker的能力就这么点那就错了,freemarker还包含数据类型、条件判断、数据遍历、内建函数等,这些功能对日常开发已经足够,看下面这段程序
package org.lxh; import java.io.File; import java.io.OutputStreamWriter; import java.io.Writer; import java.util.ArrayList; import java.util.Date; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Random; import freemarker.template.Configuration; import freemarker.template.Template; public class UseIf { public static void main(String[] args) throws Exception { //创建freemarker配置实例 Configuration config=new Configuration(); config.setDirectoryForTemplateLoading(new File("template")); //创建数据模型 Map<String,Object> m=new HashMap<String,Object>(); m.put("score", new Random().nextInt(100)); UserInfo u1=new UserInfo(); u1.setName("仓木麻衣"); u1.setPassword("123456"); UserInfo u2=new UserInfo(); u2.setName("柯南"); u2.setPassword("654321"); List<UserInfo> all=new ArrayList<UserInfo>(); all.add(u2); all.add(u1); m.put("userList", all); m.put("time", new Date()); m.put("str", "北京,上海,杭州"); //加载模板文件 Template t=config.getTemplate("useif.ftl"); //显示生成后的数据 Writer w=new OutputStreamWriter(System.out); t.process(m, w); w.close(); } }
看下模板文件的内容:
--------------if语句的使用---------------- <#if score gte 60> 及格 <#elseif score gte 80&&score lte 90> 良好 <#else> 高材生 </#if> --------------空值判断、默认值--------------- ${name!"未定义"} --------------判断值是否存在------------ <#if name??> name存在 <#else> name不存在 </#if> -------------使用list遍历数据-------------- <#list userList as user> <#if user_has_next> 最后一组:${user.name}-${user.password}<#else>${user.name}-${user.password}</#if> </#list> -------------其他内建函数----------------- (1)日期格式化 ${time?string("yyyy-MM-dd")} (2)截取字符串 ${str?substring(0,2)} (2)indexof的使用 ${str?last_index_of(",")} (2)split的使用 <#list "12,13,14,15"?split(",") as item> ${item} </#list>
对于list的遍历有两个隐含变量用的比较多分别是_index _has_next ,详细的使用可以百度一下,真的很简单
三 freemarker宏指令
宏指令可以简单理解成在freemarker可以自己定义的公共方法,具体用途主要用于分页,来看下面的这个最简单的例子
package org.lxh; import java.io.File; import java.io.IOException; import java.io.OutputStreamWriter; import java.io.Writer; import java.util.HashMap; import java.util.Map; import freemarker.template.Configuration; import freemarker.template.Template; public class UseMacro { public static void main(String[] args) throws Exception { //创建freemarker配置实例 Configuration config=new Configuration(); config.setDirectoryForTemplateLoading(new File("template")); //创建数据模型 Map<String,Object> m=new HashMap<String,Object>(); m.put("type", "other"); m.put("num1", 2); m.put("num2", 3); //加载模板文件 Template t=config.getTemplate("macro.ftl"); //显示生成后的数据 Writer w=new OutputStreamWriter(System.out); t.process(m, w); w.close(); } }
模板文件的内容如下:
----------------------宏指令的使用 m1可以看成是方法的名称,num1、num2为入参---------------------- <#macro m1 num1 num2> <#assign result=num1+num2> <h3>${result}</h3> </#macro> <@m1 5 6/> ----------------------宏指令(嵌入式)----------------- <#macro m2> <h3><#nested></h3> </#macro> <@m2>hello world</@m2>
运行效果如下:
四 其他指令
包含指令:<#include "test.txt" />
import指令:<#import "b.ftl" as bb />
<#macro copyright date> <p>Copyright (C) ${date}.</p> </#macro>
当ftl文件比较多的时候,而且这些ftl有引用关系,这个时候import就可以大展拳脚了,如<@bb.copyright date="2010-2011" />
五 struts2中使用freemarker
struts2是不能直接使用freemarker的,如果一定要用可以:
(1)解压struts2-core-X.X.X.jar文件,把在META-INF文件夹下面的struts-tags.tld文件复制到WEB-INF文件夹下。 将freemark的jar导入到工程中
(2)在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指令导入标签库,这样在freemarker中就可以使用struts2的标签了
<#assign s=JspTaglibs["/WEB-INF/struts-tags.tld"] /> 注:这里我把struts-tags.tld放在WEB-INF下面
(4)在FreeMarker模板中使用struts标签
<@s;property value=http://www.mamicode.com/""/>
注:struts使用freemarker要在result的节点加上type="freemarker"
对于这种做法有些时候不好用,freemarker也提供了在jsp页面使用freemarker的方法,同样的还是需要把freemarker的tld文件放在WEB-INF下面,这个文件如果找不到可以建一个叫fmtag.tld的文件,内容如下:
<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN" "http://java.sun.com/j2ee/dtds/web-jsptaglib_1_1.dtd"> <taglib> <tlibversion>2.0</tlibversion> <jspversion>1.1</jspversion> <shortname>FreeMarker JSP Support</shortname> <tag> <name>template</name> <tagclass>freemarker.ext.jsp.FreemarkerTag</tagclass> <bodycontent>tagdependent</bodycontent> <info>Allows evaluation of FreeMarker templates inside JSP</info> <attribute> <name>caching</name> <required>false</required> </attribute> </tag> </taglib>
接下来看看在jsp中怎么使用freemarker
(1)jsp页面引入freemarker标签库
<%@ taglib prefix="fm" uri="/WEB-INF/fmtag.tld" %>
(2)嵌入freemarker
<fm:template> freemarker语句 </fm:template>
综合前面的情况无论怎么用是不是都很简单呢
六 扩充freemarker内建函数
freemarker虽然提供很多内建函数,但仍然有需要自定义的时候,freemarker也运行用户写自定义函数,如下所示:
public class MyFmTag implements TemplateMethodModel { public Object exec(List param) throws TemplateModelException { Object result=null; if("save".equals(param.get(0).toString())){ //调用保存方法 } return result; } //保存方法................ }使用的时候就像下面这样就ok了
<#assign check= "org.lxh.util.MyFmTag"?new()/>
<#assign result=check(‘save‘,‘xxx‘)/>
freemarker的使用心得