第四章 Controller接口控制器詳解(3)

4.11、AbstractWizardFormController

嚮導控制器類提供了多步驟(嚮導)表單的支持(如完善個人資料時分步驟填寫基本信息、工作信息、學校信息等)

假設現在做一個完善個人信息的功能,分三個頁面展示:

1、頁面1完善基本信息;

2、頁面2完善學校信息;

3、頁面3完善工作信息。

這裏我們要注意的是當用戶跳轉到頁面2時頁面1的信息是需要保存起來的,還記得AbstractFormController中的sessionForm嗎? 如果爲true則表單數據存放到session中,哈哈,AbstractWizardFormController就是使用了這個特性。


嚮導中的頁碼從0開始;

PARAM_TARGET = "_target":

用於選擇嚮導中的要使用的頁面參數名前綴,如“_target0”則選擇第0個頁面顯示,即圖中的“wizard/baseInfo”,以此類推,如“_target1”將選擇第1頁面,要得到的頁碼爲去除前綴“_target”後的數字即是;

PARAM_FINISH = "_finish":

如果請求參數中有名爲“_finish”的參數,表示嚮導成功結束,將會調用processFinish方法進行完成時的功能處理;

PARAM_CANCEL = "_cancel":

如果請求參數中有名爲“_cancel”的參數,表示嚮導被取消,將會調用processCancel方法進行取消時的功能處理;

嚮導中的命令對象:

嚮導中的每一個步驟都會把相關的參數綁定到命令對象,該表單對象默認放置在session中,從而可以跨越多次請求得到該命令對象。

 

接下來具體看一下如何使用吧。

(1、修改我們的模型數據以支持多步驟提交: 

 

Java代碼  
  1. public class UserModel {  
  2.     private String username;  
  3.     private String password;  
  4.     private String realname; //真實姓名  
  5.     private WorkInfoModel workInfo;  
  6.     private SchoolInfoModel schoolInfo;  
  7.     //省略getter/setter  
  8. }  

 

Java代碼  
  1. public class SchoolInfoModel {  
  2.     private String schoolType; //學校類型:高中、中專、大學  
  3.     private String schoolName; //學校名稱  
  4.     private String specialty; //專業  
  5. //省略getter/setter  
  6. }  

 

Java代碼  
  1. public class WorkInfoModel {  
  2.     private String city; //所在城市  
  3.     private String job; //職位  
  4.     private String year; //工作年限  
  5. //省略getter/setter  
  6. }  

 

(2、控制器

  

Java代碼  
  1. package cn.javass.chapter4.web.controller;  
  2. //省略import  
  3. public class InfoFillWizardFormController extends AbstractWizardFormController {      
  4.     public InfoFillWizardFormController() {  
  5.         setCommandClass(UserModel.class);  
  6.         setCommandName("user");  
  7.     }  
  8.     protected Map referenceData(HttpServletRequest request, int page) throws Exception {  
  9.         Map map = new HashMap();  
  10.         if(page==1) { //如果是填寫學校信息頁 需要學校類型信息  
  11.             map.put("schoolTypeList", Arrays.asList("高中", "中專", "大學"));  
  12.         }  
  13.         if(page==2) {//如果是填寫工作信息頁 需要工作城市信息  
  14.             map.put("cityList", Arrays.asList("濟南", "北京", "上海"));  
  15.         }  
  16.         return map;  
  17.     }     
  18.     protected void validatePage(Object command, Errors errors, int page) {  
  19.         //提供每一頁數據的驗證處理方法  
  20.     }  
  21.     protected void postProcessPage(HttpServletRequest request, Object command, Errors errors, int page) throws Exception {  
  22.         //提供給每一頁完成時的後處理方法  
  23.     }  
  24.     protected ModelAndView processFinish(HttpServletRequest req, HttpServletResponse resp, Object command, BindException errors) throws Exception {  
  25.         //成功後的處理方法  
  26.         System.out.println(command);  
  27.         return new ModelAndView("redirect:/success");  
  28.     }  
  29.     protected ModelAndView processCancel(HttpServletRequest request, HttpServletResponse response, Object command, BindException errors) throws Exception {  
  30.         //取消後的處理方法  
  31.         System.out.println(command);  
  32.         return new ModelAndView("redirect:/cancel");  
  33.     }  
  34. }  

 

page頁碼:是根據請求中以“_target”開頭的參數名來確定的,如“_target0”,則頁碼爲0;

referenceData:提供每一頁需要的表單支持對象,如完善學校信息需要學校類型,page頁碼從0開始(而且根據請求參數中以“_target”開頭的參數來確定當前頁碼,如_target1,則page=1);

validatePage:驗證當前頁的命令對象數據,驗證應根據page頁碼來分步驟驗證;

postProcessPage:驗證成功後的後處理;

processFinish:成功時執行的方法,此處直接重定向到/success控制器(詳見CancelController);

processCancel:取消時執行的方法,此處直接重定向到/cancel控制器(詳見SuccessController);

 

其他需要了解:

allowDirtyBack和allowDirtyForward:決定在當前頁面驗證失敗時,是否允許嚮導前移和後退,默認false不允許;

onBindAndValidate(HttpServletRequest request, Object command, BindException errors, int page):允許覆蓋默認的綁定參數到命令對象和驗證流程。

(3、spring配置文件(chapter4-servlet.xml)

 

Java代碼  
  1. <bean name="/infoFillWizard"   
  2. class="cn.javass.chapter4.web.controller.InfoFillWizardFormController">  
  3.     <property name="pages">  
  4.         <list>  
  5.             <value>wizard/baseInfo</value>  
  6.             <value>wizard/schoolInfo</value>  
  7.             <value>wizard/workInfo</value>  
  8.        </list>  
  9.     </property>  
  10. </bean>      

 

pages:表示嚮導中每一個步驟的邏輯視圖名,當InfoFillWizardFormController的page=0,則將會選擇“wizard/baseInfo”,以此類推,從而可以按步驟選擇要展示的視圖。

 

(4、嚮導中的每一步視圖

(4.1、基本信息頁面(第一步) baseInfo.jsp: 

 

Java代碼  
  1. <form method="post">  
  2. 真實姓名:<input type="text" name="realname" value="${user.realname}"><br/>  
  3. <input type="submit" name="_target1" value="下一步"/>  
  4. </form>  

 

當前頁碼爲0;

name="_target1":表示嚮導下一步要顯示的頁面的頁碼爲1;

 

(4.2、學校信息頁面(第二步) schoolInfo.jsp:

 

Java代碼  
  1. <form method="post">  
  2. 學校類型:<select name="schoolInfo.schoolType">  
  3.   <c:forEach items="${schoolTypeList }" var="schoolType">  
  4.    <option value="${schoolType }"   
  5.        <c:if test="${user.schoolInfo.schoolType eq schoolType}">  
  6.            selected="selected"  
  7.        </c:if>  
  8.    >  
  9.        ${schoolType}  
  10.    </option>  
  11.   </c:forEach>  
  12. </select><br/>  
  13. 學校名稱:<input type="text" name="schoolInfo.schoolName" value="${user.schoolInfo.schoolName}"/><br/>  
  14. 專業:<input type="text" name="schoolInfo.specialty" value="${user.schoolInfo.specialty}"/><br/>  
  15. <input type="submit" name="_target0" value="上一步"/>  
  16. <input type="submit" name="_target2" value="下一步"/>  
  17. </form>  

 

(4.3、工作信息頁面(第三步) workInfo.jsp:

 

Java代碼  
  1. <form method="post">  
  2. 所在城市:<select name="workInfo.city">  
  3.   <c:forEach items="${cityList }" var="city">  
  4.    <option value="${city }"   
  5.        <c:if test="${user.workInfo.city eq city}">selected="selected"</c:if>  
  6.    >  
  7.      ${city}  
  8.    </option>  
  9.   </c:forEach>  
  10. </select><br/>  
  11. 職位:<input type="text" name="workInfo.job" value="${user.workInfo.job}"/><br/>  
  12. 工作年限:<input type="text" name="workInfo.year" value="${user.workInfo.year}"/><br/>  
  13. <input type="submit" name="_target1" value="上一步"/>  
  14. <input type="submit" name="_finish" value="完成"/>  
  15. <input type="submit" name="_cancel" value="取消"/>  
  16. </form>  

 

當前頁碼爲2;

name="_target1":上一步,表示嚮導上一步要顯示的頁面的頁碼爲1;

name="_finish":嚮導完成,表示嚮導成功,將會調用嚮導控制器的processFinish方法

name="_cancel":嚮導取消,表示嚮導被取消,將會調用嚮導控制器的processCancel方法

 

到此嚮導控制器完成,此處的嚮導流程比較簡單,如果需要更復雜的頁面流程控制,可以選擇使用Spring Web Flow框架。

 

4.12、ParameterizableViewController

參數化視圖控制器,不進行功能處理(即靜態視圖),根據參數的邏輯視圖名直接選擇需要展示的視圖。 

 

Java代碼  
  1. <bean name="/parameterizableView"   
  2. class="org.springframework.web.servlet.mvc.ParameterizableViewController">  
  3. <property name="viewName" value="success"/>  
  4. </bean>  

 

該控制器接收到請求後直接選擇參數化的視圖,這樣的好處是在配置文件中配置,從而避免程序的硬編碼,比如像幫助頁面等不需要進行功能處理,因此直接使用該控制器映射到視圖。

4.13、AbstractUrlViewController

提供根據請求URL路徑直接轉化爲邏輯視圖名的支持基類,即不需要功能處理,直接根據URL計算出邏輯視圖名,並選擇具體視圖進行展示:

urlDecode:是否進行url解碼,不指定則默認使用服務器編碼進行解碼(如Tomcat默認ISO-8859-1);

urlPathHelper:用於解析請求路徑的工具類,默認爲org.springframework.web.util.UrlPathHelper。

 

UrlFilenameViewController是它的一個實現者,因此我們應該使用UrlFilenameViewController。

 

4.14、UrlFilenameViewController

將請求的URL路徑轉換爲邏輯視圖名並返回的轉換控制器,即不需要功能處理,直接根據URL計算出邏輯視圖名,並選擇具體視圖進行展示:

 

根據請求URL路徑計算邏輯視圖名; 

 

Java代碼  
  1. <bean name="/index1/*"   
  2. class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>  
  3. <bean name="/index2/**"   
  4. class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>  
  5. <bean name="/*.html"   
  6. class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>  
  7. <bean name="/index3/*.html"   
  8. class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>  

 

/index1/*:可以匹配/index1/demo,但不匹配/index1/demo/demo,如/index1/demo邏輯視圖名爲demo;

/index2/**:可以匹配/index2路徑下的所有子路徑,如匹配/index2/demo,或/index2/demo/demo,“/index2/demo”的邏輯視圖名爲demo,而“/index2/demo/demo”邏輯視圖名爲demo/demo;

/*.html:可以匹配如/abc.html,邏輯視圖名爲abc,後綴會被刪除(不僅僅可以是html);

/index3/*.html:可以匹配/index3/abc.html,邏輯視圖名也是abc;

 

上述模式爲Spring Web MVC使用的Ant-style 模式進行匹配的:

 

Java代碼  
  1. ?    匹配一個字符,如/index? 可以匹配 /index1 , 但不能匹配 /index 或 /index12  
  2. *    匹配零個或多個字符,如/index1/*,可以匹配/index1/demo,但不匹配/index1/demo/demo 
  3. **   匹配零個或多個路徑,如/index2/**:可以匹配/index2路徑下的所有子路徑,如匹配/index2/demo,或/index2/demo/demo 
  4.  
  5. 如果我有如下模式,那Spring該選擇哪一個執行呢?當我的請求爲“/long/long”時如下所示: 
  6. /long/long 
  7. /long/**/abc  
  8. /long/** 
  9. /** 
  10. Spring的AbstractUrlHandlerMapping使用:最長匹配優先; 
  11. 如請求爲“/long/long” 將匹配第一個“/long/long”,但請求“/long/acd” 則將匹配 “/long/**”,如請求“/long/aa/abc”則匹配“/long/**/abc”,如請求“/abc”則將匹配“/**”  

 

UrlFilenameViewController還提供瞭如下屬性:

prefix:生成邏輯視圖名的前綴;

suffix:生成邏輯視圖名的後綴;

 

Java代碼  
  1. protected String postProcessViewName(String viewName) {  
  2.         return getPrefix() + viewName + getSuffix();  
  3. }  

 

Java代碼  
  1. <bean name="/*.htm" class="org.springframework.web.servlet.mvc.UrlFilenameViewController">  
  2.         <property name="prefix" value="test"/>  
  3.         <property name="suffix" value="test"/>  
  4. </bean>  

 

prefix=“test”,suffix=“test”,如上所示的/*.htm:可以匹配如/abc.htm,但邏輯視圖名將變爲testabctest。

 


私塾在線學習網
原創內容(http://sishuok.com

原創內容,轉載請註明私塾在線【http://sishuok.com/forum/blogPost/list/5254.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章