context session request关系整理
这里描述的是java web开发中的含义,三者都可以通过使用serAttribute方法保存键值对用于共享变量,只是作用域不一样,范围由大到小为context、session、request。
context
context意为“上下文”,即程序的运行环境。
- ServletContext
这个是web应用级的上下文。web容器(比如tomcat、jboss、weblogic等)启动的时候,它会为每个web应用程序创建一个ServletContext对象,它代表当前web应用的上下文(注意:是每个web应用有且仅创建一个ServletContext,一个web应用,就是你一个web工程)。一个web中的所有servlet共享一个ServletContext对象,所以可以通过ServletContext对象来实现Servlet之间的通讯。在一个继承自HttpServlet对象的类中,可以通过this.getServletContext来获取。;
public class ContextShare extends HttpServlet throws ServletException,IOException{
protected void contextVarShare(HttpServletRequest request,HttpServletResponse response) {
// 使用ServletContext共享变量
ServletContext sc = this.getServletContext();
sc.setAttribute("name", "contextName");
}
}
- ApplicationContext
在web容器启动时,会触发容器初始化事件,此时ContextLoaderListener会监听到这个事件,其contextInitialized方法会被调用,在这个方法中,spring会初始化一个启动上下文,这个上下文被称为根上下文,即WebApplicationContext,这是一个接口(这个接口继承ApplicationContext这个接口),确切的说,其实际的实现类是XmlWebApplicationContext。这个就是Spring的IoC容器,其对应的Bean定义的配置由web.xml中的context-param标签指定。在这个IoC容器初始化完毕后,Spring以WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE为属性Key,将其存储到ServletContext中,便于获取。 - servletContext:(这边用斜体表示,以防与ServletContext混淆,这边这个其实是指每个HttpServlet都有一个与自己对应的上下文)
contextLoaderListener监听器初始化完毕后,开始初始化web.xml中配置的Servlet,这个servlet可以配置多个,以最常见的DispatcherServlet为例,这个servlet实际上是一个标准的前端控制器,用以转发、匹配、处理每个servlet请求。DispatcherServlet上下文在初始化的时候会建立自己的IoC上下文,用以持有spring mvc相关的bean。在建立DispatcherServlet自己的IoC上下文时,会利用WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE先从ServletContext中获取之前的根上下文(即WebApplicationContext)作为自己上下文的parent上下文(有个parent属性作为对Spring的ApplicationContext的引用)。有了这个parent上下文之后,再初始化自己持有的上下文。这个DispatcherServlet初始化自己上下文的工作在其initStrategies方法中可以看到,大概的工作就是初始化处理器映射、视图解析等。这个servlet自己持有的上下文默认实现类也是xmlWebApplicationContext。初始化完毕后,spring以与servlet的名字相关(此处不是简单的以servlet名为Key,而是通过一些转换,具体可自行查看源码)的属性为属性Key,也将其存到ServletContext中,以便后续使用。这样每个servlet就持有自己的上下文,即拥有自己独立的bean空间,同时各个servlet共享相同的bean,即根上下文(第2步中初始化的上下文)定义的那些bean。
session
在WEB开发中,服务器可以为每个用户浏览器创建一个会话对象(session对象),注意:一个浏览器独占一个session对象(默认情况下)。因此,在需要保存用户数据时,服务器程序可以把用户数据写到用户浏览器独占的session中,当用户使用浏览器访问其它程序时,其它程序可以从用户的session中取出该用户的数据,为用户服务。
简而言之:关闭再打开浏览器,在这个情况下HttpSession、HttpServletRequest的实例都发生了改变。
public class SessionShare extends HttpServlet{
protected void sessionVarShare(HttpServletRequest request,HttpServletResponse response) {
// 使用HttpSession共享变量
HttpSession session = request.getSession();
session.setAttribute("name", "sessionName");
}
}
request
同一个页面,或者采用转发机制进行处理,ServletContext、HttpSession、HttpServletRequest的实例都不会发生改变。
重定向、超链接,在这个情况下HttpServletRequest的实例发生了变化。
public class RequestShare extends HttpServlet{
protected void requestVarShare(HttpServletRequest request,HttpServletResponse response) {
// 使用HttpServletRequest共享变量
request.setAttribute("name", "requestName");
}
}
- 转发跳转
// 使用转发跳转处理
request.getRequestDispatcher("/GetSharingVars").forward(request, response);
- 重定向
//使用重定向跳转处理
response.sendRedirect("/HelloWorld/GetSharingVars");