1. 程式人生 > >springmvc+Freemarker配置說明詳解1

springmvc+Freemarker配置說明詳解1

Freemarker使用模板技術進行檢視的渲染。自從看了Struts標籤、Freemarker、JSTL的效能對比後,我毅然決定放棄Struts標籤了!效率太差…… 
Spring本身支援了對Freemarker的整合。只需要配置一個針對Freemarker的檢視解析器即可。

二、Spring MVC檢視解析器

檢視解析器的工作流程大致是這樣的:

  1. Controller的某個方法執行完成以後,返回一個檢視(比如:listUser)
  2. 檢視解析器要做的工作就是找到某個物件來完成檢視的渲染,或者跳轉到其他的邏輯檢視。這裡的渲染物件通常就是我們的jsp檔案或者我們下面用的Freemarker(例如listUser.jsp或者listUser.ftl)。
  3. 渲染完成以後,將解析結果傳送到客戶端瀏覽器

下面介紹一下本文需要用到的解析器(更多解析器資料):

  • InternalResourceViewResolver:這是一個最常用的解析器。通常使用它指定渲染物件為jsp頁面
  • FreeMarkerViewResolver:這就是Spring與Freemarker整合需要用到的解析器

三、配置多檢視,支援freemarker

我們通常不希望所有的動態頁面請求都使用Freemarker來渲染,那就需要配置多個檢視解析器。網上有很多這方面的帖子。我看到很多人的做法是在web.xml中配置兩個DispatcherServlet,一個攔截.do,一個攔截

.ftl;然後再寫兩個dispatcherServlet.xml,配置兩個檢視解析器;jsp頁面、ftl模板就各司其職。 其實沒有那麼複雜。

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配置,否則會找不到頁面。


classic_compatible=true              ##如果變數為null,轉化為空字串,比如做比較的時候按照空字元做比較 
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元素