Spring MVC
Java Web项目一般都是基于Spring MVC 的。而Spring MVC是建立在IoC容器基础上的。我们知道,Spring MVC和Spring IoC 都是独立的模块,因此web下需要将IoC容器启动与Web容器的启动过程集成在一起。下边看下是怎么做到的。
Servlet规范
这里我们先看下 Servlet 规范,即Java Servlet Specification。也就是 JSR 340。
翻开Web 应用部分 -> 目录结构. 介绍如下:
WEB-INF目录中的内容有:/WEB-INF/web.xml : 部署描述文件。
/WEB-INF/classes/ : servlet 和实用工具类目录 /WEB-INF/classes/。此目录中的类对应用程序类加载器必须是可见的。
/WEB-INF/lib/*.jar : java 归档文件区域。这些文件中包括了servlet,bean,静态资源和打包在JAR文件中的 JSP 文件,以及其他对Web 应用程序有用的实用工具类。Web 应用程序的类加载器必须能够从这些归档文件中加载类。
Web 应用程序类加载器必须先从 WEB-INF/classes 目录下加载类,然后从 WEB-INF/lib 目录下的 JAR 库中加载。此外,除了静态资源打包在JAR 文件中的情况外,任何来自客户端的请求访问 WEB-INF/目录中的资源必须返回一个 SC_NOT_FOUND(404)的响应。
这里Servlet 规范定义了各个文件或文件夹的作用。然后翻到Web 应用部署
当一个 Web 应用程序部署到容器中,在 Web 应用程序开始处理客户端请求之前,必须按照下述步骤顺序执行。
1. 实例化部署描述文件中<listener>元素标识的每个事件监听器的一个实例。
2. 对于已实例化的实现了 ServletContextListener 接口的监听器实例,调用 contextInitialized() 方法。
3. 实例化部署描述文件中<filter>元素标识的每个过滤器的一个实例,并调用每个过滤器实例的init()方法。
4. 包含<load-on-startup>元素的<servlet>元素,根据 load-on-startup 元素值定义的顺序为每个 servlet 实例化一个实例,并调用每个 servlet 实例的 init() 方法。
有了上边的介绍,先看一个Spring MVC的web.xml 配置。
<?xml version="1.0" encoding="UTF-8"?> <web-app version="3.0" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"> <filter> <filter-name>characterEncodingFilter</filter-name> <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> <init-param> <param-name>encoding</param-name> <param-value>UTF-8</param-value> </init-param> <init-param> <param-name>forceEncoding</param-name> <param-value>true</param-value> </init-param> </filter> <filter-mapping> <filter-name>characterEncodingFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:beans.xml</param-value> </context-param> <servlet> <servlet-name>DispatcherServlet</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>DispatcherServlet</servlet-name> <url-pattern>/*</url-pattern> </servlet-mapping> </web-app>对于此 web.xml,启动Web项目时
1. WEB容器会去读取配置文件web.xml,读取<listener>和<context-param>两个结点。这里是我们的Spring配置文件。
2. 创建一个唯一的ServletContext(servlet上下文),即根上下文。这个web项目的所有部分都将共享这个上下文。
3. 容器将<context-param>转换为键值对,并交给servletContext
4. 实例化一个ContextLoaderListener 实例,并调用contextInitialized() 方法。
5. 实例化一个CharacterEncodingFilter 实例,并调用其init() 方法。
6. 实例化一个DispatcherServlet实例,并调用其init() 方法。(加了<load-on-startup>标签)
简单来说,加载顺序为:context-param -> listener -> filter -> servlet。
ServletContext
在一个Web应用中,改对象是唯一的。可以简单将ServletContext看作一个Web应用的服务器端组件的共享内存。这点可以参考它的方法:
方法 | m描述 |
setAttribute(String name, Object object) | 把一个对象和属性名绑定,并将这个对象存放在ServletContext中 |
getAttribute(String name) | 根据给定的属性名返回所绑定的对象 |
removeAttribute(String name) | 根据给定的属性名从ServletContext中删除相应的属性 |
getAttributeNames() | 返回一个Enumeration对象,包含了存储在ServletContext对象中所有的属性名 |
ContextLoaderListener
这个是Spring web项目的常用配置,看其contextInitialized() 方法,主要是在父类 ContextLoader 中,主要做了两件事:1. 创建一个WebApplicationContext,并将其放入ServletContext 中。
2. 加载双亲容器。
参考:
https://www.gitbook.com/book/waylau/servlet-3-1-specification/details