springmvc+Freemarker配置說明詳解1
Freemarker使用模板技術進行檢視的渲染。自從看了Struts標籤、Freemarker、JSTL的效能對比後,我毅然決定放棄Struts標籤了!效率太差……
Spring本身支援了對Freemarker的整合。只需要配置一個針對Freemarker的檢視解析器即可。
二、Spring MVC檢視解析器
檢視解析器的工作流程大致是這樣的:
- Controller的某個方法執行完成以後,返回一個檢視(比如:listUser)
- 檢視解析器要做的工作就是找到某個物件來完成檢視的渲染,或者跳轉到其他的邏輯檢視。這裡的渲染物件通常就是我們的jsp檔案或者我們下面用的Freemarker(例如listUser.jsp或者listUser.ftl)。
- 渲染完成以後,將解析結果傳送到客戶端瀏覽器
下面介紹一下本文需要用到的解析器(更多解析器資料):
- InternalResourceViewResolver:這是一個最常用的解析器。通常使用它指定渲染物件為jsp頁面
- FreeMarkerViewResolver:這就是Spring與Freemarker整合需要用到的解析器
三、配置多檢視,支援freemarker
我們通常不希望所有的動態頁面請求都使用Freemarker來渲染,那就需要配置多個檢視解析器。網上有很多這方面的帖子。我看到很多人的做法是在web.xml中配置兩個DispatcherServlet,一個攔截.do,一個攔截
1.Web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<!-- ======================================================== -->
<!-- Spring MVC Config Servlet -->
<!-- ======================================================== -->
<!-- JSP DispatcherServlet -->
<servlet>
<servlet-name>dispatcher</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/classes/applicationContext.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<!-- ======================================================== -->
<!-- Spring MVC Config Mapping -->
<!-- ======================================================== -->
<servlet-mapping>
<servlet-name>dispatcher</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>
</web-app>
2.dispatcherServlet.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">
<!--通用檢視解析器-->
<bean id="viewResolverCommon" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/page/"/>
<property name="suffix" value=".jsp"/><!--可為空,方便實現自已的依據副檔名來選擇檢視解釋類的邏輯 -->
<property name="viewClass">
<value>org.springframework.web.servlet.view.InternalResourceView</value>
</property>
<property name="order" value="1"/>
</bean>
<!-- 配置freeMarker檢視解析器 -->
<bean id="viewResolverFtl" class="org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver">
<property name="viewClass" value="org.springframework.web.servlet.view.freemarker.FreeMarkerView"/>
<property name="contentType" value="text/html; charset=utf-8"/>
<property name="cache" value="true" />
<property name="suffix" value=".ftl" />
<property name="order" value="0"/>
</bean>
<!-- 配置freeMarker的模板路徑 -->
<bean id="freemarkerConfig" class="org.springframework.web.servlet.view.freemarker.FreeMarkerConfigurer">
<property name="templateLoaderPath">
<value>/WEB-INF/ftl/</value>
</property>
<property name="freemarkerVariables">
<map>
<entry key="xml_escape" value-ref="fmXmlEscape" />
</map>
</property>
<property name="defaultEncoding">
<value>utf-8</value>
</property>
<property name="freemarkerSettings">
<props>
<prop key="template_update_delay">3600</prop>
</props>
</property>
</bean>
<bean id="fmXmlEscape" class="freemarker.template.utility.XmlEscape"/>
<!-- 註解支援 -->
<mvc:annotation-driven/>
<!-- 對包中的所有類進行掃描,以完成Bean建立和自動依賴注入的功能 -->
<context:component-scan base-package="com.hl.usersmanager">
<!-- 允許定義過濾器將基包下的某些類納入或排除
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/> -->
</context:component-scan>
</beans>
3.Controller
package com.hl.usersmanager.controller;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.ModelMap;
import org.springframework.web.bind.annotation.RequestMapping;
import com.hl.usersmanager.model.Users;
import com.hl.usersmanager.service.IUserService;
@Controller
public class UserController {
// 使用註解實現自動裝配 不需要再寫get set方法以及在context中配置bean
@Autowired
private IUserService userService;
@RequestMapping(value = "findUserByName.do")
public String findUserByName(String name,ModelMap model) {
Users users = userService.findUserByName(name);
model.addAttribute("userPhone",users.getPhone());
System.out.println("userPhone:" + users.getPhone());
return "showUser";
}
@RequestMapping(value = "findAllUsers.do")
public String findAllUsers(ModelMap model) {
List<Users> users = userService.findAllUsers();
model.addAttribute("users",users);
return "listUser";
}
……
}
在檢視解析器中有一個<property name="order" value="orderValue"/>
的配置,這個配置表示解析器的優先級別。我們將FreeMarkerViewResolver
的級別設為0
,將InternalResourceViewResolver
的級別設為1
。這樣,解析器就會優先使用 FreeMarkerViewResolver
進行解析,如果找不到相應的模板,就使用InternalResourceViewResolver
進行解析,如果還找不到頁面,就會產生一個404
錯誤!
在本例中,我們在/WEB-INF/page/
下有一個showUser.jsp
頁面,在/WEB-INF/ftl/
下有一個listUser.ftl
的模板檔案。那麼當訪問findAllUsers.do
的時候,Controller
返回一個listUser
檢視,根據解析器配置,先使用FreeMarkerViewResolver
進行解析。它會根據模板路徑templateLoaderPath
的配置/WEB-INF/ftl/
下去找是否有一個listUser
並且以後綴配置suffix
值.ftl
,即listUser.ftl
檔案,如果找到則使用該模板進行解析。這裡我們實現已經建立了這個模板檔案,所以user列表成功被顯示出來。
當用戶訪問findUserByName.do
的時候,返回showUser
檢視,毅然先使用FreeMarkerViewResolver
進行解析,結果發現在/WEB-INF/ftl/
下並沒有showUser.ftl
這個模板檔案,於是使用InternalResourceViewResolver
進行解析,於是開始尋找/WEB-INF/page/
下是否有showUser.jsp
檔案。由於我們已經建立了這個檔案,於是最終使用showUser.jsp
進行渲染。那麼如果沒有找到showUser.jsp
,就會丟擲404
錯誤。
這裡還要注意的是,如果Controller
中返回檢視加了字尾jsp
或者ftl
,在配置中就不要加入suffix
配置,否則會找不到頁面。
whitespace_stripping=true ##去掉多餘的空格,非常有用
##模版更新事件,設定為1秒,正式環境設定為3600秒
#template_update_delay=3600
template_update_delay=1 ##模版更新時間,這裡配置是1秒更新一次,正式環境,模版不會改變,可以將這個值設很大,提高效率.就是被這個引數害死了.老是不更新.
locale=zh_CN ##中國
default_encoding=utf-8 ##編碼utf8
url_escaping_charset=utf-8 ##url編碼utf8
date_format=yyyy-MM-dd ##顯示日期格式
time_format=HH:mm:Ss ##顯示時間格式
datetime_format=yyyy-MM-dd ##顯示日期格式
number_format=\#0.\#\#\#\#\# ##數字顯示格式hi.
#template_update_delay=1 datetime_format=yyyy-MM-dd HH:mm:ss date_format=yyyy-MM-dd time_format=HH:mm:ss number_format=0.######; boolean_format=true,false auto_import="/common/index.ftl" as ui whitespace_stripping=true default_encoding=UTF-8 tag_syntax=auto_detect url_escaping_charset=UTF-8
裡面大部分都不解釋了,就解釋一個, auto_import="/common/index.ftl" as ui 這個表示每個freemarker的檢視頁面都會自動引入這個ftl檔案。裡面定義的就是一些巨集,如text文字框,各種form元素