使用Java進(jìn)行FreeMarker的web模板開發(fā)的基礎(chǔ)教程
一、概述
FreeMarker 是一個(gè)模板引擎,一個(gè)基于模板生成文本輸出的通用工具,使用純 Java 編寫,F(xiàn)reeMarker 被設(shè)計(jì)用來生成 HTML Web 頁面,特別是基于 MVC 模式的應(yīng)用程序,雖然 FreeMarker 具有一些編程的能力,但通常由 Java 程序準(zhǔn)備要顯示的數(shù)據(jù),由FreeMarker 生成頁面,通過模板顯示準(zhǔn)備的數(shù)據(jù)(如下圖)

FreeMarker 不是一個(gè) Web 應(yīng)用框架,而適合作為 Web 應(yīng)用框架一個(gè)組件。FreeMarker 與容器無關(guān),因?yàn)樗⒉恢?HTTP 或 Servlet;FreeMarker 同樣可以應(yīng)用于非Web應(yīng)用程序環(huán)境,F(xiàn)reeMarker 更適合作為 Model2 框架(如 Struts)的視圖組件,你也可以在模板中使用 JSP標(biāo)記庫。另外,F(xiàn)reeMarker是免費(fèi)的。
二、Freemarker的準(zhǔn)備條件
freemarker.2.3.16.jar,下載地址這里就不貼了..(這個(gè)jar包其實(shí)在struts2里面)
三、Freemarker生成靜態(tài)頁面的原理
Freemarker 生成靜態(tài)頁面,首先需要使用自己定義的模板頁面,這個(gè)模板頁面可以是最最普通的html,也可以是嵌套freemarker中的 取值表達(dá)式, 標(biāo)簽或者自定義標(biāo)簽等等,然后后臺(tái)讀取這個(gè)模板頁面,解析其中的標(biāo)簽完成相對(duì)應(yīng)的操作, 然后采用鍵值對(duì)的方式傳遞參數(shù)替換模板中的的取值表達(dá)式,做完之后 根據(jù)配置的路徑生成一個(gè)新的html頁面, 以達(dá)到靜態(tài)化訪問的目的。
四、Freemarker提供的標(biāo)簽
Freemarker提供了很多有用 常用的標(biāo)簽,F(xiàn)reemarker標(biāo)簽都是<#標(biāo)簽名稱>這樣子命名的,${value} 表示輸出變量名的內(nèi)容 ,具體如下:
1、list:該標(biāo)簽主要是進(jìn)行迭代服務(wù)器端傳遞過來的List集合,比如:
<#list nameList as names>
${names}
</#list>
name是list循環(huán)的時(shí)候取的一個(gè)循環(huán)變量,freemarker在解析list標(biāo)簽的時(shí)候,等價(jià)于:
for (String names : nameList) {
System.out.println(names);
}
2、if:該標(biāo)簽主要是做if判斷用的,比如:
<#if (names=="陳靖仇")> 他的武器是: 十五~~ </#if>
這個(gè)是條件判斷標(biāo)簽,要注意的是條件等式必須用括號(hào)括起來, 等價(jià)于:
if(names.equals("陳靖仇")){
System.out.println("他的武器是: 十五~~");
}
3、include:該標(biāo)簽用于導(dǎo)入文件用的。
<#include "include.html"/>
這個(gè)導(dǎo)入標(biāo)簽非常好用,特別是頁面的重用。
另外在靜態(tài)文件中可以使用${} 獲取值,取值方式和el表達(dá)式一樣,非常方便。
下面舉個(gè)例子(static.html):
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
描述:${description}
<br/>
集合大小:${nameList?size}
<br/>
迭代list集合:
<br/>
<#list nameList as names>
這是第${names_index+1}個(gè)人,叫做:<label style="color:red">${names}</label>
if判斷:
<br/>
<#if (names=="陳靖仇")>
他的武器是: 十五~~
<#elseif (names=="宇文拓")> <#--注意這里沒有返回而是在最后面-->
他的武器是: 軒轅劍~·
<#else>
她的絕招是:蠱毒~~
</#if>
<br/>
</#list>
迭代map集合:
<br/>
<#list weaponMap?keys as key>
key--->${key}<br/>
value----->${weaponMap[key]!("null")}
<#--
fremarker 不支持null, 可以用! 來代替為空的值。
其實(shí)也可以給一個(gè)默認(rèn)值
value-----${weaponMap[key]?default("null")}
還可以 在輸出前判斷是否為null
<#if weaponMap[key]??></#if>都可以
-->
<br/>
</#list>
include導(dǎo)入文件:
<br/>
<#include "include.html"/>
</body>
</html>
實(shí)際代碼:
package com.chenghui.test;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapper;
import freemarker.template.Template;
import freemarker.template.TemplateException;
public class CreateHtml {
public static void main(String[] args) {
try {
//創(chuàng)建一個(gè)合適的Configration對(duì)象
Configuration configuration = new Configuration();
configuration.setDirectoryForTemplateLoading(new File("D:\\project\\webProject\\WebContent\\WEB-INF\\template"));
configuration.setObjectWrapper(new DefaultObjectWrapper());
configuration.setDefaultEncoding("UTF-8"); //這個(gè)一定要設(shè)置,不然在生成的頁面中 會(huì)亂碼
//獲取或創(chuàng)建一個(gè)模版。
Template template = configuration.getTemplate("static.html");
Map<String, Object> paramMap = new HashMap<String, Object>();
paramMap.put("description", "我正在學(xué)習(xí)使用Freemarker生成靜態(tài)文件!");
List<String> nameList = new ArrayList<String>();
nameList.add("陳靖仇");
nameList.add("玉兒");
nameList.add("宇文拓");
paramMap.put("nameList", nameList);
Map<String, Object> weaponMap = new HashMap<String, Object>();
weaponMap.put("first", "軒轅劍");
weaponMap.put("second", "崆峒印");
weaponMap.put("third", "女媧石");
weaponMap.put("fourth", "神農(nóng)鼎");
weaponMap.put("fifth", "伏羲琴");
weaponMap.put("sixth", "昆侖鏡");
weaponMap.put("seventh", null);
paramMap.put("weaponMap", weaponMap);
Writer writer = new OutputStreamWriter(new FileOutputStream("success.html"),"UTF-8");
template.process(paramMap, writer);
System.out.println("恭喜,生成成功~~");
} catch (IOException e) {
e.printStackTrace();
} catch (TemplateException e) {
e.printStackTrace();
}
}
}
這樣子基本上可以算的上可以簡(jiǎn)單的去做一點(diǎn)簡(jiǎn)單的生成了,但是要在實(shí)際中去運(yùn)用,還是差的很遠(yuǎn)的,因?yàn)閒reemarker給的標(biāo)簽完全滿足不了我們的需要,這時(shí)候就需要自定義標(biāo)簽來完成我們的需求了。。
五、Freemarker自定義標(biāo)簽
Freemarker自定義標(biāo)簽就是自己寫標(biāo)簽,然后自己解析,完全由自己來控制標(biāo)簽的輸入輸出,極大的為程序員提供了很大的發(fā)揮空間。
基于步驟:
以前寫標(biāo)簽需要在<后加# ,但是freemarker要識(shí)別自定義標(biāo)簽需要在后面加上@,然后后面可以定義一些參數(shù),當(dāng)程序執(zhí)行template.process(paramMap, out);,就會(huì)去解析整個(gè)頁面的所有的freemarker標(biāo)簽。
自定義標(biāo)簽 需要自定義一個(gè)類,然后實(shí)現(xiàn)TemplateDirectiveModel,重寫execute方法,完成獲取參數(shù),根據(jù)參數(shù)do something等等。。
將自定義標(biāo)簽與解析類綁定在一起需要在paramMap中放入該解析類的實(shí)例,存放的key與自定義標(biāo)簽一致即可。。
注意:在自定義標(biāo)簽中,如果標(biāo)簽內(nèi)什么也沒有,開始標(biāo)簽和結(jié)束標(biāo)簽絕對(duì)不能再同一行,不然會(huì)報(bào)錯(cuò)
freemarker.log.JDK14LoggerFactory$JDK14Logger error
我曾經(jīng)上當(dāng)過,這是freemarker 存在的bug。
下面是static.html的例子:
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<#--自定義變量-->
<#assign num='hehe'/>
${num}
<br/>
自定義標(biāo)簽
<@content name="chenghui" age="120">
${output}
${append}
</@content>
</body>
</html>
下面是上面的static.html模板的解析類:
package com.chenghui.test;
import static freemarker.template.ObjectWrapper.DEFAULT_WRAPPER;
import java.io.IOException;
import java.io.Writer;
import java.util.Map;
import freemarker.core.Environment;
import freemarker.template.TemplateDirectiveBody;
import freemarker.template.TemplateDirectiveModel;
import freemarker.template.TemplateException;
import freemarker.template.TemplateModel;
import freemarker.template.TemplateModelException;
import freemarker.template.TemplateNumberModel;
import freemarker.template.TemplateScalarModel;
/**
* 自定義標(biāo)簽解析類
* @author Administrator
*
*/
public class ContentDirective implements TemplateDirectiveModel{
private static final String PARAM_NAME = "name";
private static final String PARAM_AGE = "age";
@Override
public void execute(Environment env, Map params,TemplateModel[] loopVars,
TemplateDirectiveBody body) throws TemplateException, IOException {
if(body==null){
throw new TemplateModelException("null body");
}else{
String name = getString(PARAM_NAME, params);
Integer age = getInt(PARAM_AGE, params);
//接收到參數(shù)之后可以根據(jù)做具體的操作,然后將數(shù)據(jù)再在頁面中顯示出來。
if(name!=null){
env.setVariable("output", DEFAULT_WRAPPER.wrap("從ContentDirective解析類中獲得的參數(shù)是:"+name+", "));
}
if(age!=null){
env.setVariable("append", DEFAULT_WRAPPER.wrap("年齡:"+age));
}
Writer out = env.getOut();
out.write("從這里輸出可以再頁面看到具體的內(nèi)容,就像document.writer寫入操作一樣。<br/>");
body.render(out);
/*
如果細(xì)心的話,會(huì)發(fā)現(xiàn)頁面上是顯示out.write()輸出的語句,然后再輸出output的內(nèi)容,
可見 在body在解析的時(shí)候會(huì)先把參數(shù)放入env中,在頁面遇到對(duì)應(yīng)的而來表單時(shí)的才會(huì)去取值
但是,如果該表單時(shí)不存在,就會(huì)報(bào)錯(cuò), 我覺得這里freemarker沒有做好,解析的時(shí)候更加會(huì)把錯(cuò)誤暴露在頁面上。
可以這樣子彌補(bǔ)${output!"null"},始終感覺沒有el表達(dá)式那樣好。
*/
}
}
/**
* 獲取String類型的參數(shù)的值
* @param paramName
* @param paramMap
* @return
* @throws TemplateModelException
*/
public static String getString(String paramName, Map<String, TemplateModel> paramMap) throws TemplateModelException{
TemplateModel model = paramMap.get(paramName);
if(model == null){
return null;
}
if(model instanceof TemplateScalarModel){
return ((TemplateScalarModel)model).getAsString();
}else if (model instanceof TemplateNumberModel) {
return ((TemplateNumberModel)model).getAsNumber().toString();
}else{
throw new TemplateModelException(paramName);
}
}
/**
*
* 獲得int類型的參數(shù)
* @param paramName
* @param paramMap
* @return
* @throws TemplateModelException
*/
public static Integer getInt(String paramName, Map<String, TemplateModel> paramMap) throws TemplateModelException{
TemplateModel model = paramMap.get(paramName);
if(model==null){
return null;
}
if(model instanceof TemplateScalarModel){
String str = ((TemplateScalarModel)model).getAsString();
try {
return Integer.valueOf(str);
} catch (NumberFormatException e) {
throw new TemplateModelException(paramName);
}
}else if(model instanceof TemplateNumberModel){
return ((TemplateNumberModel)model).getAsNumber().intValue();
}else{
throw new TemplateModelException(paramName);
}
}
}
然后再前面的實(shí)際代碼中加上:
//自定義標(biāo)簽解析
paramMap.put("content", new ContentDirective());
這樣子基本上可以使用,freemarker完成自定義標(biāo)簽了,解決一寫簡(jiǎn)單的業(yè)務(wù)邏輯, 但是在實(shí)際的項(xiàng)目中不可能這樣子去做,因?yàn)檫€沒有和spring進(jìn)行集成使用,每次都需要在解析的時(shí)候把解析類的實(shí)例放進(jìn)去。。
- java Spring整合Freemarker的詳細(xì)步驟
- Java操作FreeMarker模板引擎的基本用法示例小結(jié)
- 基于Java的Spring框架來操作FreeMarker模板的示例
- java Freemarker頁面靜態(tài)化實(shí)例詳解
- Java實(shí)現(xiàn)用Freemarker完美導(dǎo)出word文檔(帶圖片)
- 基于Freemarker和xml實(shí)現(xiàn)Java導(dǎo)出word
- JAVA集成Freemarker生成靜態(tài)html過程解析
- Java超級(jí)實(shí)用的Freemarker工具類
- 在Java中FreeMarker?模板來定義字符串模板
- Java使用Freemarker頁面靜態(tài)化生成的實(shí)現(xiàn)
相關(guān)文章
利用Spring JPA中的@Version注解實(shí)現(xiàn)樂觀鎖
樂觀鎖是數(shù)據(jù)庫和應(yīng)用程序中使用的一種并發(fā)控制策略,用于在多個(gè)事務(wù)嘗試更新單個(gè)記錄時(shí)確保數(shù)據(jù)完整性,Java Persistence API (JPA) 提供了一種借助@Version注解在 Java 應(yīng)用程序中實(shí)現(xiàn)樂觀鎖的機(jī)制,文中有詳細(xì)的代碼示例供大家參考,需要的朋友可以參考下2023-11-11
深入解讀 Spring Boot 生態(tài)之功能、組件與優(yōu)勢(shì)
本文將深入剖析 Spring Boot 的生態(tài)體系,包括其核心功能、生態(tài)組件以及在不同場(chǎng)景中的應(yīng)用,并附上一張 Spring Boot 生態(tài)系統(tǒng)圖,幫助開發(fā)者更直觀地理解 Spring Boot 的強(qiáng)大之處,感興趣的朋友一起看看吧2024-11-11
通過@Resource注解實(shí)現(xiàn)屬性裝配代碼詳解
這篇文章主要介紹了通過@Resource注解實(shí)現(xiàn)屬性裝配代碼詳解,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01
Java中JDom解析XML_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
JDOM是一種解析XML的Java工具包。DOM適合于當(dāng)今流行的各種語言,包括Java,JavaScripte,VB,VBScript,Perl,C,C++等。下面通過本文給大家介紹Java中JDom解析XML的方法,感興趣的朋友一起學(xué)習(xí)吧2017-07-07
Java編程構(gòu)造方法與對(duì)象的創(chuàng)建詳解
這篇文章主要介紹了Java編程構(gòu)造方法與對(duì)象的創(chuàng)建詳解,具有一定參考價(jià)值,需要的朋友可以了解下。2017-11-11
Java使用ReentrantLock進(jìn)行加解鎖的示例代碼
在多線程編程中,為了確保多個(gè)線程在訪問共享資源時(shí)不會(huì)發(fā)生沖突,我們通常需要使用 鎖 來同步對(duì)資源的訪問,本文將深入探討如何優(yōu)雅地使用 ReentrantLock,避免常見的坑點(diǎn),并提升代碼的可維護(hù)性,需要的朋友可以參考下2025-04-04

