最近参加面试,简历上写了Spring,SpringMVC。然后面试官死磕这部分,当时记忆模糊,当场就hold不住了!事后,我觉得有必要重新翻阅一下SpringMVC的整个执行流程,写这系列文章,算是对自己的一个小惩罚吧,但为了梦想更进一步,我没啥怨言!
先来一张大图做个引导,这系列文章基本也是按照这个图进行说明:
1.ContextLoaderListner
将SpringMVC,必须先从web.xml开始讲起,从配置ContextLoaderListner开始,图中所示:
平时我们直接使用Spring做项目的时候是通过ClassPathXmlApplicationContext这种方式去加载Spring的配置文件,而对于web项目不同,我们得想办法将配置文件的路径以context-param的方式加载并且使用ContextLoaderListner进行监听读取;ContextLoaderListner的作用就是在启动web容器的时候,自动装配ApplicationContext的配置信息,看一眼ContextLoaderListner的构造你就明白了;
从图中可知,ContextLoaderListner实现了ServletContextListener这个接口,而每个应用都有一个ServletContext的全局对象与之关联,这个ServletContext对象在应用启动时被创建,在程序关闭时被销毁;因此,提前剧透下,ServletContextListener的核心就是初始化WebApplicationContext实例,并将其放值在ServletContext这个全局对象中。
这个地方,可以深入想一下,如果要定义我们自己的ContextListner类,我们完全可以实现ServletContextListener或者继承ContextLoaderListner来做。
言归正传,回到ContextLoaderListner。ServletContext启动之后会调用ServletContextListener#contextInitialized,因此又回到了下图中所示内容:
介绍下WebApplicationContext,他其实是ApplicationContext的子接口,为SpringMVC设计的,如图所示:
继续往下走,看ContextLoader#initWebApplicationContext实现,说下它的实现逻辑,代码咋们一步一步贴:
1.首先判端web.xml中是否多次定义了ContextLoaderListner,若是,则直接抛异常;
2.创建WebApplicationContext的实例;
在这个地方要初始化我们的根上下文,它会通过读取ContextLoader类的目录下属性文件ContextLoader.properties,并根据其中的配置实现WebApplicationContext的实现类
ContextLoader.properties中内容如下所示:
3.将webApplicationContext实例set进servletContext中,让其跟着servletContext同生共死;
4.获取当前线程对于的ClassLoader,并且将其与创建的context实例放入Map中,做映射;
以上便是对ContextLoaderListner中逻辑的一个讲解,其实说到底ContextLoaderListner就做了一件事,初始化WebApplicationContext的具体实例作为IOC容器,关于DispatchServlet的讲解请看后面更新;