关键词:Spring MVC, DispatcherServlet前端控制器, HandlerMapping处理器映射器, HandlAdapter处理器适配器, ViewResolver视图解析器, Handler处理器, View resolver视图解析器, View视图。
1、Spring MVC 工作原理图
Spring MVC,底层基于servlet,通过实现Servlet接口的DispatcherServlet类封装其核心功能;将前端传来的请求,查询处理器映射器,调用对应的处理程序;同时实现里对视图解析、本地语言、主题解析以及上传下载文件等功能的支持。默认的处理程序是非常简单的Controller接口,只有一个ModelAndView hangleRequest(request, response)方法。Spring提供了一个控制器层次结构,可以派生子类。如果应用程序需要处理用户输入表单,那么可以继承AbstractFormController。如果应用程序需要处理用户输入表单,那么可以继承AbstractWizardFormController。
2、Spring MVC流程
1)客户端发送请求,至前端控制器DispatcherServlet;
2) DispatcherServlet收到请求,并调用HandlerMapping处理器映射器;
3) HandlerMapping处理器映射器,按照xml配置或注解进行查找,找到具体的处理器,生成处理器对象及处理器拦截器一并返回给DispatcherServlet;
4)DispatcherServlet调用HandlerAdapter处理器适配器;
5) HandlerAdapter经过适配调用具体的处理器(Controller,也叫后端控制器);
6)Controller执行完成返回ModelAndView;
7)HandlerAdapter将controller执行结果ModelAndView返回给DispatcherServlet;
8)DispatcherServlet将ModelAndView传给ViewReslover视图解析器;
9)ViewReslover解析后返回具体View;
10)DispatcherServlet根据View进行渲染视图(即将模型数据填充至视图中);
11)DispatcherServlet响应客户端。
3、组件详解:
DispatcherServlet前端控制器,是整个流程控制的中心,控制其它组件执行,统一调度,降低组件之间的耦合性,提高每个组件的扩展性。
HandlerMapping处理器映射器,通过扩展处理器映射器,实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等。
HandlAdapter处理器适配器,通过扩展处理器适配器,支持更多类型的处理器。
ViewResolver视图解析器,通过扩展视图解析器,支持更多类型的视图解析,例如:jsp、freemarker、pdf、excel等。
组件:
1、DispatcherServlet前端控制器
作用:接收请求,响应结果,相当于转发器,中央处理器。有了dispatcherServlet减少了其它组件之间的耦合度。
用户请求到达前端控制器,它就相当于mvc模式中的c,dispatcherServlet是整个流程控制的中心,由它调用其它组件处理用户的请求,dispatcherServlet的存在降低了组件之间的耦合性。
2、HandlerMapping处理器映射器
作用:根据请求的url查找Handler
HandlerMapping负责根据用户请求找到Handler即处理器,springmvc提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等。
3、HandlerAdapter处理器适配器
作用:按照特定规则(HandlerAdapter要求的规则)去执行Handler
通过HandlerAdapter对处理器进行执行,这是适配器模式的应用,通过扩展适配器可以对更多类型的处理器进行执行。
4、Handler处理器(需开发)
注意:编写Handler时按照HandlerAdapter的要求去做,这样适配器才可以去正确执行Handler
Handler 是继DispatcherServlet前端控制器的后端控制器,在DispatcherServlet的控制下Handler对具体的用户请求进行处理。
由于Handler涉及到具体的用户业务请求,所以一般情况需要工程师根据业务需求开发Handler。
5、View resolver视图解析器
作用:进行视图解析,根据逻辑视图名解析成真正的视图(view)
View Resolver负责将处理结果生成View视图,View Resolver首先根据逻辑视图名解析成物理视图名即具体的页面地址,再生成View视图对象,最后对View进行渲染将处理结果通过页面展示给用户。 springmvc框架提供了很多的View视图类型,包括:jstlView、freemarkerView、pdfView等。
一般情况下需要通过页面标签或页面模版技术将模型数据通过页面展示给用户,需要由工程师根据业务需求开发具体的页面。
6、View视图(需开发)
View是一个接口,实现类支持不同的View类型(jsp、freemarker、pdf...)
核心架构的具体流程步骤如下:
1、首先用户发送请求——>DispatcherServlet,前端控制器收到请求后自己不进行处理,而是委托给其他的解析器进行处理,作为统一访问点,进行全局的流程控制;
2、DispatcherServlet——>HandlerMapping, HandlerMapping 将会把请求映射为HandlerExecutionChain 对象(包含一个Handler 处理器(页面控制器)对象、多个HandlerInterceptor 拦截器)对象,通过这种策略模式,很容易添加新的映射策略;
3、DispatcherServlet——>HandlerAdapter,HandlerAdapter 将会把处理器包装为适配器,从而支持多种类型的处理器,即适配器设计模式的应用,从而很容易支持很多类型的处理器;
4、HandlerAdapter——>处理器功能处理方法的调用,HandlerAdapter 将会根据适配的结果调用真正的处理器的功能处理方法,完成功能处理;并返回一个ModelAndView 对象(包含模型数据、逻辑视图名);
5、ModelAndView的逻辑视图名——> ViewResolver, ViewResolver 将把逻辑视图名解析为具体的View,通过这种策略模式,很容易更换其他视图技术;
6、View——>渲染,View会根据传进来的Model模型数据进行渲染,此处的Model实际是一个Map数据结构,因此很容易支持其他视图技术;
7、返回控制权给DispatcherServlet,由DispatcherServlet返回响应给用户,到此一个流程结束。
下边两个组件通常情况下需要开发:
Handler:处理器,即后端控制器用controller表示。
View:视图,即展示给用户的界面,视图中通常需要标签语言展示模型数据。
看到这些步骤我相信大家很感觉非常的乱,这是正常的,但是这里主要是要大家理解springMVC中的几个组件:
DispatcherServlet前端控制器:接收请求,响应结果;
HandlerMapping处理器映射器:根据URL查找处理器;
Handler处理器:开发实现;
HandlerAdapter处理器适配器:会把处理器包装成适配器,这样就可以支持多种类型的处理器,类比笔记本的适配器(适配器模式的应用);
ViewResovler视图解析器:进行视图解析,多返回的字符串,进行处理,可以解析成对应的页面。
4、实现
web.xml
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>org.springframework.web.context.request.RequestContextListener</listener-class>
</listener>
<servlet>
<servlet-name>action</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:*-mvc.xml</param-value>
</init-param>
<!-- 通过初始化参数指定springmvc配置文件的位置 -->
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>action</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
4.1、ContextLoaderListener
对于SpringMVC功能的实现分析。我们首先从web.xml开始,在web.xml文件中我们首先配置的就是ContextLoaderListener。
ContextLoaderListener的作用启动web容器时,自动装配ApplicationContext的配置信息(spring.xml)。因为他实现了ServletContextListener这个接口,在web.xml配置这个监听器,启动容器时,就会默认执行它实现的方法,使用ServletContextListener这个接口,开发者能够在为客户端请求提供服务之前向ServletContext中添加任意的对象。这个对象在ServletContext启动的时候被初始化,然后在ServletContext整个运行期间都是可见的。每一个Web应用都有一个ServletContext与之相关联。ServletContext对象在引用启动时被创建,在引用关闭时被销毁。ServletContext在全局范围内有效,类似于应用中的一个全局变量。
1.1、ServletContextListener
(1)自定义实现ServletContextListener,首先我们创建ServletContextListener,目标是在系统启动时添加自定义的属性,以便于在全局范围内可以随时调用。系统启动的时候会调用ServletContextListener实现类的contextInitialized方法,所以需要在这个方法中实现我们的初始化逻辑:
(2)注册监听器:在web.xml文件中需要注册自定义的监听器:
(3)启动tomcat服务器,我们就能在任意的servlet或者jsp中通过下面的方式来获取我们的初始化参数:
1.2、ContextLoaderListener
ContextLoaderListener实现了ServletContextListener接口,因此在ServletContext启动之后会调用ServletContextListener的contextInitialized方法,那么,我们就从这个函数开始进行分析:
这里涉及了一个常用类WebApplicationContext;在web应用中,我们会用到WebApplicationContext,WebApplicationContext继承自ApplicationContext,在ApplicationContext的基础上又追加了一些特定于Web的操作及属性。其主要实现如下:
(1)检验WebApplicationContext存在性,
(2)创建WebApplicationContext实例,
(3)将实例记录在serveltContext中,
(4)映射当前的类加载器与创建的实例到全局变量currentContextPerThread中。
2、DispatcherServlet
在Spring中,ContextLoaderListener只是辅助功能,用于创建WebApplicationContext类型实例,而真正的逻辑实现其实就是在DispatcherServlet中进行的,DispatcherServlet是实现servlet接口的实现类。servlet是一个java编写的程序,此程序是基于HTTP协议的,在服务器端运行的,是按照servelt规范编写的一个java类。servelt主要是处理客户端的请求并将其结果发送到客户端。Servelt的框架是由两个java包组成,javax.servelt和javax.servlet.http。javax.servlet定义了所有的servlet类都必须实现或扩展的通用接口和类,在javax.servle.http定义了采用HTTP通信协议的HttpServlet类。
2.1、DispatcherServlet的初始化(init() )
DispatcherServlet的初始化过程主要是通过将当前的servlet类型实例转换为BeanWrapper类型实例,以便使用Spring中提供的注入功能进行对应的属性的注入。属性注入包括:
1)封装及验证初始化参数,
2)将当前servlet实例转化成BeanWrapper实例,
3)注册相对于Resource的属性编辑器,
4)属性注入,
5)servletBean的初始化 。
在ContextLoaderListener加载的时候已经创建了WebApplicationContext实例,而在这个函数中最重要的就是对这个实例进一步的补充初始化:
protected final void initServletBean() throws ServletException {
Logger... //日志处理
//对WebApplication进行补充处理
this.webApplicationContext = initWebApplicationContext();
Logger... //日志处理
}
2.2、WebApplicationContext的初始化
initWebApplicationContext函数的主要工作就是创建或刷新WebApplicationContext实例并对servlet功能所使用的变量进行初始化。其主要包括以下几个部分:
1)寻找或创建对应的WebApplicationContext实例
2)调用configureAndRefreshWebApplicationContext方法来对已经创建的WebApplicationContext实例进行配置及刷新
3)刷新:onRefresh是FrameworkServlet类中提供的模板方法,在其子类DispathcerServlet中进行了重写,主要用于刷新Spring在web功能实现中所必须使用的全局变量。
参考: