springmvc的ModelAttribute注解

        SpringMVC中@ModelAttribute是一个重要的属性,理解好它是用好自动封装等功能的基础。但是@ModelAttribute和@RequestMapping组合后的情况较多,很容易让人糊涂。今天我在博客中总结一下常见的情况。

一、基本作用:向Model中put对象属性

例:一个没有使用@ModelAttribute的Controller方法.

[java]  view plain
  1. @RequestMapping("/save")  
  2. public String save(User user) {  
  3.     user.setUsername("U love me");  
  4.     userService.save(user);  
  5.     return "result";  
  6. }  
其中User包含id和username两个私有属性,含有公共setter和getter方法.
执行此方法时会将key为"user"(注意:这里即使参数名称是user1,key一样还是"user"),value为user的对象加入到model.在jsp页面可以通过${user.id}和${user.name}得到值的,即上面方法和下面方法是相当的.
[java]  view plain
  1. @RequestMapping("/save")  
  2. public String save(Model model,int id,String username) {  
  3.     User user=new User();  
  4.     //这里是通过反射从request里面拿值再set到user  
  5.     user.setId(id);  
  6.     user.setUsername(username);  
  7.     model.addAttribute("user",user);  
  8.       
  9.     user.setUsername("U love me");  
  10.     userService.save(user);  
  11.     return "result";  
  12. }  

二、@ModelAttribute的基本用法

1.在方法上使用@ModelAttribute

   在方法上使用@ModelAttribute("k1"),结果是向Model中添加一个键值对,key是"k1",value是方法的返回值。


[java]  view plain
  1. @ModelAttribute("user1")  
  2. public User addUser(User user) {  
  3.     return new User(520,"I love U");  
  4. }  
假设此方法是写在UserController内,那么执行UserController内带有@RequestMapping的方法之前,都会先执行此addUser方法.并且执行addUser过程中会添加两个对象到model,先将key为"user"的对象(由addUser方法的User user引起的),再添加key为"user1"的对象(由注解@ModelAttribute("user1")引起的).

2.在方法参数上使用@ModelAttribute.

 在方法参数上使用@ModelAttribute("k1"),结果是向Model中添加一个键值对,key是"k1",value是其后的对象。

[java]  view plain
  1. @RequestMapping("/save")  
  2. public String save(@ModelAttribute("userA") User user) {  
  3.     user.setUsername("U love me");  
  4.     userService.save(user);  
  5.     return "result";  
  6. }  
此方法会先从model去获取key为"userA"的对象,如果获取不到会通过反射实例化一个User对象,再从request里面拿值set到这个对象,然后把这个User对象添加到model(其中key为"userA").
使用了@ModelAttribute可以不在括号中指定key(框架默认指定一个key),此情况下,用与不用@ModelAttribute没有区别.


3.在方法和方法参数上结合使用@ModelAttribute,即上面两步的两个方法都添加ModelAttribute,如下:

[java]  view plain
  1. @ModelAttribute("user1")  
  2. public User addUser(User user) {  
  3.     return new User(520,"I love U");  
  4. }  
  5. @RequestMapping("/save")  
  6. public String save(@ModelAttribute User user) {  
  7.     user.setUsername("U love me");  
  8.     userService.save(user);  
  9.     return "result";  
  10. }  
假设要执行保存用户操作,根据一分析可知,先执行完会产生两个User类型的对象(一个key是"user",另一个key是"user1")添加到model,再执行save方法,此时会先从model去找key为"user"的对象,能找到再从request取值set到这个User对象.最后返回到jsp页面,model里也只有两个User类型对象.
再来个小小假设,将上面@ModelAttribute("user1")的user1改为user,其它不变.虽然执行了addUser方法,那么执行到save方法内,user对象的字段值还是来源于请求,最后返回到jsp页面,model里也只有一个User类型对象.


4.另类:@ModelAttribute和@RequestMapping同时使用

@Controller
@RequestMapping("/hello/")
public class HelloWorldContorller{
	@ModelAttribute  
	@RequestMapping("/saveA")  
	public String save(@ModelAttribute User user) {  
		user.setUsername("U love me");  
		userService.save(user);  
		return "result";  
	}  
}


此种情况,会添加一个key为"user"的User对象到model,还会添加一个key为"string",value为"result"的对象到model,而视图名称则变为了"hello/saveA"而不是"result".相信一般开发者都不会这样用.

三.下面再看看结合@SessionAttributes的用法.

1.先看看这段代码

[java]  view plain
  1. @Controller  
  2. @RequestMapping("user")  
  3. @SessionAttributes("test")  
  4. public class UserController {  
  5.     @RequestMapping(value = "test1")  
  6.     public String test1(Map<String, Object> model, SessionStatus sessionStatus) {  
  7.         model.put("test","something");  
  8.         //sessionStatus.setComplete();  
  9.         return "user/list";  
  10.     }  
  11.   
  12.     @RequestMapping(value = "test2")  
  13.     public String test2(Map<String,?>  model) {  
  14.         Object test = model.get("test");  
  15.         System.out.println(test);  
  16.         return "user/list";  
  17.     }  
  18. }  
在类级别上使用@SessionAttributes("test"),它的作用是在controller共享 model 属性,直到调用org.springframework.web.bind.support.SessionStatus#setComplete会清除此session值.否则长期保留.因此先调用/user/test1,再调用/user/test2.在test2方法能得到test1方法put进去的值.当移除上面的注解,因为属性为test的session也被清除了,所以test2方法会得到是null.

2.再看结合@ModelAttribute的用法.

[java]  view plain
  1. @Controller  
  2. @RequestMapping("user")  
  3. @SessionAttributes("test")  
  4. public class UserController {  
  5.     @RequestMapping(value = "test1")  
  6.     public String test1(HttpServletRequest request) {  
  7.         User user = new User("xiejx618");  
  8.         request.getSession().setAttribute("test",user);  
  9.         return "user/list";  
  10.     }  
  11.     @RequestMapping(value = "test2")  
  12.     public String test2(@ModelAttribute("test") User user,SessionStatus sessionStatus) {  
  13.         System.out.println(user.getUsername());  
  14.         sessionStatus.setComplete();  
  15.         return "user/list";  
  16.     }  
  17. }  
一样先调用/user/test1,再调用/user/test2.在test2方法能得到test1方法set进去的值.使用了@SessionAttributes,如果在controller的方法参数上有@ModelAttribute,那么此方法会确保属性test的session会存在,否则会抛org.springframework.web.HttpSessionRequiredException异常,即上面代码先不调/user/test1,直接调/user/test2就会抛此异常


我个人很少这样用,用原始的写法就清晰很多了,但要看明白别人写的代码。

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值