深入理解springMVC中的Model和Session属性
作者:吉姆怀特
作为一个javaweb应用的开发者,你快速学习了request(HttpRequest)和Session(HttpSession)的范围,理解这些范围并且在这些范围内数据和对象是如何是进出的对设计和构建web应用是非常关键的。
springMVC的范围
当我用springMVC写web应用的时候,我发现spring model和session有一点神秘—特别是与http reques、和session范围关联起来这些我都已经了解了。
spring的model元素会在我的session或者request中找到吗,如果是这样,我怎么控制这个过程呢,在这篇文章中我希望能够解密springMVC的model和session是如何工作的。
spring的@MODELATTRIBUTE
这里有好几种向spring的Model添加数据的方式。数据或者对象通常通过在controller上的注释方法添加到spring中的model中去。
下边这个例子中,@ModelAttribute用来将MyCommandBean的实例以key值为myRequestObject添加到model中去
@Controller public class MyController { @ModelAttribute("myRequestObject") public MyCommandBean addStuffToRequestScope() { System.out.println("Inside of addStuffToRequestScope"); MyCommandBean bean = new MyCommandBean("Hello World",42); return bean; } @RequestMapping("/dosomething") public String requestHandlingMethod(Model model, HttpServletRequest request) { System.out.println("Inside of dosomething handler method"); System.out.println("--- Model data ---"); Map modelMap = model.asMap(); for (Object modelKey : modelMap.keySet()) { Object modelValue = modelMap.get(modelKey); System.out.println(modelKey + " -- " + modelValue); } System.out.println("=== Request data ==="); java.util.Enumeration reqEnum = request.getAttributeNames(); while (reqEnum.hasMoreElements()) { String s = reqEnum.nextElement(); System.out.println(s); System.out.println("==" + request.getAttribute(s)); } return "nextpage"; } // ... the rest of the controller }
在一个请求的request中,任何使用@ModelAttribute注解的方法会在controller的handler方法(像上边例子汇总的requestHandlingMethod 方法)之前被调用。
在这些handler方法执行前,这些方法把数据增加到java.util.map中最终添加到spring Model中去。这可以通过一个简单的实验证明,我创建了两个jsp页面:index.jsp和nextpage.jsp。
index.jsp中的链接用来发送request到web应用中来触发Mycontroller中的requestHandlingMethod()方法。requestHandlingMethod()方法返回“nextpage”作为下一个视图逻辑上的名字,在这个例子中我们解析为nextpage.jsp。
当这个小的web站点用这种方式执行的时候,controller里边的System.out.println方法表明了@ModelAttribute方法是如何在handler方法之前运行的。
它同样也展示了这个MyCommandBean被创建和添加到springModel中去的过程。
Inside of addStuffToRequestScope Inside of dosomething handler method --- Model data --- myRequestObject -- MyCommandBean [someString=Hello World, someNumber=42] === Request data === org.springframework.web.servlet.DispatcherServlet.THEME_SOURCE ==WebApplicationContext for namespace 'dispatcher-servlet': startup date [Sun Oct 13 21:40:56 CDT 2013]; root of context hierarchy org.springframework.web.servlet.DispatcherServlet.THEME_RESOLVER ==org.springframework.web.servlet.theme.FixedThemeResolver@204af48c org.springframework.web.servlet.DispatcherServlet.CONTEXT ==WebApplicationContext for namespace 'dispatcher-servlet': startup date [Sun Oct 13 21:40:56 CDT 2013]; root of context hierarchy org.springframework.web.servlet.HandlerMapping.pathWithinHandlerMapping ==dosomething.request org.springframework.web.servlet.HandlerMapping.bestMatchingPattern ==/dosomething.* org.springframework.web.servlet.DispatcherServlet.LOCALE_RESOLVER ==org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver@18fd23e4
现在的问题是“springModel数据存储在哪?”它存储在标准的java request范围中吗?答案是“是的”,从上边的输出可以看出,当handler方法执行的时候MyCommandBean是在model中,但是没有在request对象中。
确实,spring不会把model数据作为request的属性,直到执行handler方法之后和下一个视图之前(在这个例子中是nextpage.jsp)
这也可以通过打印存储在index.jsp和nextpage.jsp中的HttpServletRequest中的数据展示出来,这两个页面我都使用jsp来展示HttpServletRequest的属性
Request Scope (key==values) <% java.util.Enumeration<String> reqEnum = request.getAttributeNames(); while (reqEnum.hasMoreElements()) { String s = reqEnum.nextElement(); out.print(s); out.println("==" + request.getAttribute(s)); %><br /> <% } %>
当应用打开并且index.jsp展现的时候,你可以看到在Request范围内没有属性
do something Request Scope(key==values)
在这个例子中,当“do something”连接被点击的时候触发了MyController的handler方法执行,继而导致nextpage.jsp被执行,考虑到这是同样的jsp内容,再次提出不免有些啰嗦。当nextpage.jsp提出的时候,表明model的MyCommandBean在controller里被创建,并已经被加到HttpServletRequest范围中去了。这个spring model的属性key已经被复制,并且当做Request属性的key。
所以spring model数据的创建要早于handler方法的执行,在下一个视图加载前就已经被复制到HttpServletRequest中去了。
spring Model和Request后边的原因
你可能会疑惑为什么spring使用model属性,为什么不直接将数据加到Request对象中去呢,在Rod Johnson的书中我找到了这个问题的答案。下边就是来自这本书的关于model元素的文本。
直接给HttpServletRequest(或者Request属性)增加元素看起来实现了相同的目的,做这件事的原因很明确,当我们查看我们为MVC框架设置的要求的时候,应该尽可能使视图无关的,意味着视图技术不和HttpServletRequest绑定。
Spring的@SESSIONATTRIBUTE
现在你知道spring的model数据是如何管理的并且是如何和Httprequset属性关联的,那么spring的session数据呢?
spring的@SessionAttributes在controller上使用指定model属性应该存储在Session中。实际上,精确的讲spring开发文档已经表明了@SessionAttributes注解“列出了应该存储在Session中或者对话存储中model属性的名字”。
实际上,@SessionAttribute允许你做的是在加载视图之前,告诉spring你的哪一个model Attributes将会被复制到httpSession中去。
Session Scope (key==values) <% java.util.Enumeration<String> sessEnum = request.getSession() .getAttributeNames(); while (sessEnum.hasMoreElements()) { String s = sessEnum.nextElement(); out.print(s); out.println("==" + request.getSession().getAttribute(s)); %><br /> <% } %>
我在MyController中用@SessionAttributes做注解,来把同样的model属性加到spring Session中去。
@Controller @SessionAttributes("myRequestObject") public class MyController { ... }
我同样也在handler方法中增加了代码来展示什么属性在httpSession中
@SuppressWarnings("rawtypes") @RequestMapping("/dosomething") public String requestHandlingMethod(Model model, HttpServletRequest request, HttpSession session) { System.out.println("Inside of dosomething handler method"); System.out.println("--- Model data ---"); Map modelMap = model.asMap(); for (Object modelKey : modelMap.keySet()) { Object modelValue = modelMap.get(modelKey); System.out.println(modelKey + " -- " + modelValue); } System.out.println("=== Request data ==="); java.util.Enumeration<String> reqEnum = request.getAttributeNames(); while (reqEnum.hasMoreElements()) { String s = reqEnum.nextElement(); System.out.println(s); System.out.println("==" + request.getAttribute(s)); } System.out.println("*** Session data ***"); Enumeration<String> e = session.getAttributeNames(); while (e.hasMoreElements()){ String s = e.nextElement(); System.out.println(s); System.out.println("**" + session.getAttribute(s)); } return "nextpage"; }
现在,当我们使用@SessionAttributes注解后我们可以看到什么在session对象中,包括springMVC处理一个http 请求的前中后的过程里。结果如下,首先是index.jsp展示,我们可以看到在HttpServletRequest和httpSession中都没有属性数据。
do something Request Scope(key == values) Session Scope(key == values)
在handler方法执行过程中(HttpServletRequest),你可以看到MyCommandBean被加入到spring model属性中去,但是没有在HttpServletRequest和HttpSession范围中。
但是在handler方法执行之后并且nextpage.jsp被加载,你可以看到model的属性数据确实被作为属性复制到HttpServletRequest和httpSession中。
控制对话属性
现在你已经对spring model和Session属性数据是如何加载到HttpServletReq。uest和httpSession有很好的理解了。但你依旧对管理spring Session中的数据心生疑惑。spring提供了移除spring Session属性的方式,也可以在HttpSession中移除。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。