SpringMVC框架 第5章 處理響應(yīng)數(shù)據(jù)
第5章 處理響應(yīng)數(shù)據(jù)
5.1 SpringMVC 輸出模型數(shù)據(jù)概述
5.1.1提供了以下幾種途徑輸出模型數(shù)據(jù)
- ModelAndView: 處理方法返回值類型為 ModelAndView 時(shí), 方法體即可通過該對(duì)象添加模型數(shù)據(jù)
- Map 及 Model: 入?yún)?springframework.ui.Model、
org.springframework.ui.ModelMap 或 java.uti.Map 時(shí),處理方法返回時(shí),Map 中的數(shù)據(jù)會(huì)自動(dòng)添加到模型中。
- @SessionAttributes: 將模型中的某個(gè)屬性暫存到 HttpSession 中,以便多個(gè)請(qǐng)求之間可以共享這個(gè)屬性
- @ModelAttribute: 方法入?yún)?biāo)注該注解后, 入?yún)⒌膶?duì)象就會(huì)放到數(shù)據(jù)模型中
5.2處理模型數(shù)據(jù)之 ModelAndView
5.2.1?ModelAndView介紹
- 控制器處理方法的返回值如果為 ModelAndView, 則其既包含視圖信息,也包含模型
數(shù)據(jù)信息。
2)添加模型數(shù)據(jù):
MoelAndView addObject(String attributeName, Object attributeValue)
ModelAndView addAllObject(Map<String, ?> modelMap)
3)設(shè)置視圖:
void setView(View view)
void setViewName(String viewName)
5.2.2 實(shí)驗(yàn)代碼
- 增加控制器方法
- 斷點(diǎn)調(diào)試
5.2.2 源碼解析
5.3 處理模型數(shù)據(jù)之 Map
5.3.1?Map介紹
1)Spring MVC 在內(nèi)部使用了一個(gè) org.springframework.ui.Model 接口存儲(chǔ)模型數(shù)據(jù)
具體使用步驟
2)Spring MVC 在調(diào)用方法前會(huì)創(chuàng)建一個(gè)隱含的模型對(duì)象作為模型數(shù)據(jù)的存儲(chǔ)容器。
3)如果方法的入?yún)?Map 或 Model 類型,Spring MVC 會(huì)將隱含模型的引用傳遞給這些入?yún)ⅰ?/p>
4)在方法體內(nèi),開發(fā)者可以通過這個(gè)入?yún)?duì)象訪問到模型中的所有數(shù)據(jù),也可以向模型中添加新的屬性數(shù)據(jù)
5.3.2 實(shí)驗(yàn)代碼
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?4)顯示結(jié)果截圖
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?5)? 注意問題:Map集合的泛型,key為String,Value為Object,而不是String??
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 6) 測(cè)試參數(shù)類型
? ?7)? 類層次結(jié)構(gòu)
? 8)? 推薦:Map, ?便于框架移植。
? 9)? 源碼參考
5.4 處理模型數(shù)據(jù)之 SessionAttributes 注解【了解】
- 若希望在多個(gè)請(qǐng)求之間共用某個(gè)模型屬性數(shù)據(jù),則可以在控制器類上標(biāo)注一個(gè) @SessionAttributes, Spring MVC 將在模型中對(duì)應(yīng)的屬性暫存到 HttpSession 中。
- @SessionAttributes 除了可以通過屬性名指定需要放到會(huì)話中的屬性外,還可以通過模型屬性的對(duì)象類型指定哪些模型屬性需要放到會(huì)話中
例如:
- @SessionAttributes(types=User.class) 會(huì)將隱含模型中所有類型為 User.class 的屬性添加到會(huì)話中。
- @SessionAttributes(value={“user1”, “user2”})
- @SessionAttributes(types={User.class, Dept.class})
- @SessionAttributes(value={“user1”, “user2”}, types={Dept.class})?
5.4.1 @SessionAttributes 源碼
5.4.2 實(shí)驗(yàn)代碼
5.5 @ModelAttribute注解
5.5.1 @ModelAttribute注解之使用場(chǎng)景【了解】 ?
5.5.2@ModelAttribute注解之示例代碼
? ?1)? 在方法定義上使用 @ModelAttribute 注解:Spring MVC 在調(diào)用目標(biāo)處理方法前,會(huì)先逐個(gè)調(diào)用在方法級(jí)上標(biāo)注了 @ModelAttribute 的方法。
? ?2)? 在方法的入?yún)⑶笆褂?@ModelAttribute 注解:可以從隱含對(duì)象中獲取隱含的模型數(shù)據(jù)中獲取對(duì)象,再將請(qǐng)求參數(shù)綁定到對(duì)象中,再傳入入?yún)?/p>
? 3)? 將方法入?yún)?duì)象添加到模型中
? 4)? 示例代碼
- 頁面表單
- 增加@ModelAttribute注解
- 測(cè)試
- 異常
//org.springframework.web.HttpSessionRequiredException: Session attribute 'user' required - not found in session
//出現(xiàn)這個(gè)異常,是@SessionAttributes(value={"user"},types={String.class})導(dǎo)致的,去掉類上的這個(gè)注解
5.5.3 @ModelAttribute源碼參考
@Target({ElementType.PARAMETER, ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface ModelAttribute { /** ?* The name of the model attribute to bind to. ?* <p>The default model attribute name is inferred from the declared ?* attribute type (i.e. the method parameter type or method return type), ?* based on the non-qualified class name: ?* e.g. "orderAddress" for class "mypackage.OrderAddress", ?* or "orderAddressList" for "List<mypackage.OrderAddress>". ?*/ String value() default "";? } |
5.5.4 @ModelAttribute注解之運(yùn)行原理
- 執(zhí)行@ModelAttribute注解所修飾的方法,將從數(shù)據(jù)庫中獲取的對(duì)象存放到Map集合中,key為user
- SpringMVC從Map集合中獲取 user對(duì)象,將表單數(shù)據(jù)封裝到與參數(shù)名稱對(duì)應(yīng)的user對(duì)象屬性上
- SpringMVC將user對(duì)象作為參數(shù),傳遞給目標(biāo)方法。
- 注意:@ModelAttribute 注解修飾的方法中,放入到Map集合中的key值,應(yīng)該和目標(biāo)方法參數(shù)類型的類名稱首字母小寫一致。
5.5.5 @ModelAttribute注解之源碼分析
- 調(diào)用 @ModelAttribute 注解修飾的方法. 實(shí)際上把 @ModelAttribute 方法中 Map 中的數(shù)據(jù)放在了 implicitModel 中.
- 解析請(qǐng)求處理器的目標(biāo)參數(shù), 實(shí)際上該目標(biāo)參數(shù)來自于 WebDataBinder 對(duì)象的 target 屬性
- 創(chuàng)建 WebDataBinder 對(duì)象
- 確定 objectName 屬性: 若傳入的 attrName 屬性值為 "", 則 objectName 為類名第一個(gè)字母小寫.
???????????注意: attrName. 若目標(biāo)方法的 POJO 屬性使用了 @ModelAttribute 來修飾,
則 attrName 值即為 @ModelAttribute 的 value 屬性值 ??
- 確定 target 屬性:
- 在 implicitModel 中查找 attrName 對(duì)應(yīng)的屬性值.
- 若存在, ok
- 若不存在: 則驗(yàn)證當(dāng)前Handler 是否使用了@SessionAttributes 進(jìn)行修飾,
- 若使用了, 則嘗試從 Session 中獲取 attrName 所對(duì)應(yīng)的屬性值.
- 若 session 中沒有對(duì)應(yīng)的屬性值, 則拋出了異常.
- 若Handler 沒有使用 @SessionAttributes 進(jìn)行修飾, 或 @SessionAttributes 中沒有使用 value 值指定的 key和 attrName 相匹配, 則通過反射創(chuàng)建了 POJO 對(duì)象 ?
3)??SpringMVC 把表單的請(qǐng)求參數(shù)賦給了 WebDataBinder 的 target 對(duì)應(yīng)的屬性.
4) ?SpringMVC 會(huì)把 WebDataBinder 的 attrName 和 target 給到 implicitModel. 進(jìn)
?而傳到 request 域?qū)ο笾?
- 把 WebDataBinder 的 target 作為參數(shù)傳遞給目標(biāo)方法的入?yún)?
5.5.6 SpringMVC 確定目標(biāo)方法 POJO 類型入?yún)⒌倪^程
- 確定一個(gè) key:
- 若目標(biāo)方法的 POJO 類型的參數(shù)木有使用 @ModelAttribute 作為修飾,
??則 key 為 POJO 類名第一個(gè)字母的小寫
- 若使用了@ModelAttribute 來修飾, 則 key 為 @ModelAttribute 注解的
??value 屬性值.
- 在 implicitModel 中查找 key 對(duì)應(yīng)的對(duì)象, 若存在, 則作為入?yún)魅?/li>
???????若在 @ModelAttribute 標(biāo)記的方法中在 Map 中保存過, 且 key 和 1) 確定的 key
???一致, 則會(huì)獲取到.
- 若 implicitModel 中不存在 key 對(duì)應(yīng)的對(duì)象, 則檢查當(dāng)前的 Handler 是否使用 @SessionAttributes 注解修飾,
- 若使用了該注解, 且 @SessionAttributes 注解的 value 屬性值中包含了 key, 則會(huì)從 HttpSession 中來獲取 key 所對(duì)應(yīng)的 value 值, 若存在則直接傳入到目標(biāo)方法的入?yún)⒅? 若不存在則將拋出異常.
- 若 Handler 沒有標(biāo)識(shí) @SessionAttributes 注解或 @SessionAttributes 注解的 value 值中不包含 key, 則會(huì)通過反射來創(chuàng)建 POJO 類型的參數(shù), 傳入為目標(biāo)方法的參數(shù)
- SpringMVC 會(huì)把 key 和 POJO 類型的對(duì)象保存到 implicitModel 中, 進(jìn)而會(huì)保存到 request 中.
5.5.7@ModelAttribute注解修飾POJO類型的入?yún)?/b>
5.5.8@sessionAttributes注解引發(fā)的異常
- 由@SessionAttributes引發(fā)的異常
- 如果在處理類定義處標(biāo)注了@SessionAttributes(“xxx”),則嘗試從會(huì)話中獲取該屬性,并將其賦給該入?yún)ⅲ缓笤儆谜?qǐng)求消息填充該入?yún)?duì)象。如果在會(huì)話中找不到對(duì)應(yīng)的屬性,則拋出 HttpSessionRequiredException 異常