基本介紹
數(shù)據(jù)回顯:模型數(shù)據(jù)導(dǎo)向視圖(模型數(shù)據(jù) ---> Controller ---> 視圖)
說明:SpringMVC在調(diào)用方法前會創(chuàng)建一個隱含的模型對象,作為模型數(shù)據(jù)的存儲容器(隱含模型)
一、ModelAndView
1
2
3
4
5
6
7
8
9
|
@RequestMapping (method = RequestMethod.POST) public ModelAndView createUser(User user) { userService.createUser(user); ModelAndView mav = new ModelAndView(); mav.setViewName( "user/createSuccess" ); mav.addObject( "user" , user); return mav; } |
二、@ModelAttribute
1、如果是用在方法的入?yún)⑸?/p>
1
2
3
4
5
6
|
@RequestMapping (value = "/handle1" ) public String handle1( @ModelAttribute ( "user" ) User user) { user.setUserId( "1000" ); return "/user/createSuccess" ; } |
處理機制如下:
A.將HTTP請求參數(shù)綁定到User對象中
B.以user為鍵,將User對象放到數(shù)據(jù)模型中 --->相當(dāng)于map.put("user", user);
2、如果是定義在方法之上
1
2
3
4
5
6
7
8
9
10
11
12
|
@ModelAttribute ( "user" ) public User getUser(){ User user = new User(); user.setUserId( "1001" ); return user; } @RequestMapping (value = "/handle2" ) public String handle2( @ModelAttribute ( "user" ) User user) { user.setUserName( "tom" ); return "/user/showUser" ; } |
處理機制如下:
- 調(diào)用目標方法前,掃描控制器中所有方法上包含@ModelAttribute的方法,并將方法的返回值放到模型數(shù)據(jù)中 (類似于上述的第二步)
- 執(zhí)行指定的方法,如果入?yún)⒂蠤ModelAttribute,則將(1)中放置于模型數(shù)據(jù)中的對象賦值給入?yún)⒌慕壎▽ο?/li>
- 根據(jù)HTTP請求繼續(xù)對綁定對象進行填充和覆蓋,得到一個整合版的對象,覆蓋模型數(shù)據(jù)中的原對象
三、Map Model(org.springframework.ui.Model和java.util.Map)
SpringMVC一旦發(fā)現(xiàn)處理方法有Map或Model類型的入?yún)ⅲ蜁㈦[含模型中對象的引用傳給這些入?yún)?/p>
1
2
3
4
5
6
7
8
|
@RequestMapping (value = "/handle3" ) public String handle3(ModelMap modelMap) { modelMap.addAttribute( "testAttr" , "value1" ); User user = (User)modelMap.get( "user" ); user.setUserName( "tom" ); return "/user/showUser" ; } |
四、SessionAttributes(SpringMVC會將模型中對應(yīng)的屬性暫存到HttpSession中)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
@Controller @RequestMapping ( "/user" ) @SessionAttributes (“user”) // ① 將②處的模型屬性自動保存到HttpSession中 public class UserController { @RequestMapping (value = "/handle4" ) public String handle4( @ModelAttribute (“user”) User user) // ② { user.setUserName( "John" ); return "redirect:/user/handle5.html" ; } @RequestMapping (value = "/handle5" ) public String handle5(ModelMapmodelMap,SessionStatussessionStatus) { User user = (User)modelMap.get(“user”); // ③讀取模型中的數(shù)據(jù) if (user != null ) { user.setUserName( "Jetty" ); sessionStatus.setComplete(); // ④讓Spring MVC清除本處理器對應(yīng)的會話屬性 } return "/user/showUser" ; } } |
上面在向handle4發(fā)送請求時,拋異常,因為@ModelAttribute 和 @SessionAttributes的處理遵循一個流程:
1、SpringMVC 在調(diào)用處理方法前,在請求線程中自動創(chuàng)建一個隱含的模型對象
2、調(diào)用所有方法上標注了@ModelAttribute的方法,并將返回值添加到隱含模型對象中
3、查看Session中是否存在@SessionAttributes("XXX")所指定的XXX屬性,如果有,將其添加到隱含對象中,此時若隱含對象已經(jīng)存在了XXX屬性,則會覆蓋掉
4、對于入?yún)俗⒘薂ModelAttribute(XXX)的方法,處理流程如下:
(1) 如果隱含模型中擁有了XXX屬性,將其賦值給入?yún)ⅲ賹⒄埱笙⑻畛湓搶ο螅缓蠓祷兀蝗绻[含模型中不存在XXX,則執(zhí)行(2)
(2) 如果XXX是會話屬性,則嘗試從從會話中取出該屬性,將其復(fù)制給入?yún)⒃賹⒄埱笙⑻畛湓搶ο蟆H绻麜拰ο笾姓也坏絏XX屬性,則拋出HttpSessionRequiredException異常;如果隱含模型和會話中均不在XXX,則執(zhí)行(3)
(3) 創(chuàng)建入?yún)ο蟮膶嵗儆谜埱笙⑻畛湓搶ο蟆?/p>
由于上面的例子不滿足4(2),所以拋出異常,那么可以定義一個帶方法級@ModelAttribute的方法,那樣就不會執(zhí)行到4(2),執(zhí)行到4(1)就已經(jīng)找到了需要的數(shù)據(jù)。
SpringMVC 的數(shù)據(jù)流圖總結(jié)
1.HTTP請求報文到達服務(wù)器,服務(wù)器將其封裝成HTTPServletRequest對象
2.SpringMVC框架截獲這個HTTPServletRequest對象
3.SpringMVC 創(chuàng)建一個隱含模型對象,作為處理本次請求的上下文數(shù)據(jù)存放處
4.SpringMVC 將一個HTTPServletRequest對象數(shù)據(jù)綁定到處理方法的入?yún)ο笾校ū韱?命令對象)
5.將綁定錯誤信息、校驗錯誤信息都保存到隱含模型中
6.本次請求的對象模型數(shù)據(jù)放到HTTPServletRequest的屬性列表中,暴露給視圖對象
7.視圖對象對已經(jīng)放在HTTPServletRequest屬性列表的模型數(shù)據(jù)進行渲染
8.把渲染后的HTTP響應(yīng)報文發(fā)送給客戶端
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。
原文鏈接:http://www.jianshu.com/p/68684b4e22fe?utm_source=tuicool&utm_medium=referral