理解springMVC中的Model和Session属性 - Go语言中文社区

理解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的属性
<hr />
<h3>Request Scope (key==values)</h3>
<%
	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中去。
<h3>Session Scope (key==values)</h3>
<%
  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中移除。
原文链接:http://www.intertech.com/Blog/understanding-spring-mvc-model-and-session-attributes/
 
 

美团招聘,有意向的私信我

岗位名称:美团到家事业群-外卖技术部-JAVA高级工程师
职级要求:P2-3~P3-2

岗位职责:
- 负责上单业务系统的开发工作,深入理解业务痛点,通过技术手段满足并促进业务快速发展需要;
- 参与系统需求分析与设计,快速响应业务需求,负责完成核心代码编写,接口规范制定;
- 参与解决项目中的重难点问题,针对复杂业务模型能够给出合理的解决方案;

岗位基本要求:
- 参与过大型复杂分布式互联网系统的设计,要求有复杂业务的开发经验和较强的逻辑/概率思维能力,善于分析、归纳、描述、沟通和解决问题;
- 本科及以上学历,扎实的计算机基础,4年及以上工作经验,长期使用Java及开源框架进行项目开发,并有一定的项目管理经验;
- 深入使用Java,熟悉掌握常用的Java类库及框架,如多线程、并发处理、I/O与网络通讯等,对Java虚拟机有较深了解;
- 擅长使用开源框架,在某个领域内对开源框架有一定的审美,做过选型研究,解读过部分或者全部源码,掌握实现原理;
- 精通 MySQL 应用开发,熟悉数据库原理和常用性能优化技术,以及 NoSQL,Queue 的原理、使用场景以及限制;
- 研究过 http 协议、搜索引擎、缓存、jvm 调优、序列化、nio、RPC 调用框架等,并且有相关实践经验;
- 有强烈的责任心和团队合作精神,良好的抗压能力,心态积极,能主动融入团队;

具备以下者优先:
- 参与过相关系统建设;
- 有高并发系统经验;
- 参与过开源工作;
- 有代码洁癖、有极客精神;

 
 
版权声明:本文来源CSDN,感谢博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://blog.csdn.net/maoyeqiu/article/details/49719437
站方申明:本站部分内容来自社区用户分享,若涉及侵权,请联系站方删除。
  • 发表于 2021-05-17 03:53:40
  • 阅读 ( 525 )
  • 分类:

0 条评论

请先 登录 后评论

官方社群

GO教程

猜你喜欢