Spring MVC 工作原理
spring mvc从接收请求到返回响应的流程
-
客户端发起request请求,dispatcherservlet接收用户请求。
-
dispatcherservlet请求handlermapping查找用户请求对应的handler(处理器/控制器)。
-
handlermapping将查找到的handler返回给dispatcherservlet。
-
dispatcherservlet请求handleradapter调用handler,handler返回modelandview给handleradpter。
-
handleradapter将接收到的modelandview返回给dispatcherservlet。
-
dispatcherservlet请求viewresolver对接收到的逻辑视图进行解析,得到view对象。
-
viewresolver将解析出来的view对象返回给dispatcherservlet。
-
dispatcherservlet使用model中的数据对view进行渲染。
-
dispatcherservlet返回最终结果给客户端。
在handleradapter调用handler过程中,还会进行一些处理:
-
httpmessageconverter:将请求信息(如json,xml等)转换为对象,并将对象转换为指定的响应信息。
-
数据转换:对请求信息进行转换,比如string转换为integer。
-
数据格式化:对请求信息进行格式化。字符串转换为日期等。
-
数据校验:验证请求数据的有效性,并将验证结果存储到error中。
dispatcherservlet前端控制器
dispatcherservlet的作用就是接收用户的请求,然后给用户响应结果。它的作用相当于一个转发器或*处理器,用于控制整个流程的执行,对各个组件进行统一调度,以降低组件的耦合性,有利于组件之间的扩展。
dispatcherservlet将会装配如下组件:
-
本地化解析器:本地化解析,允许一个实例。
-
主题解析器:主题解析,只允许一个实例。
-
处理器映射器:请求到处理器的映射,允许多个实例。
-
处理器适配器:执行handler,允许多个实例。
-
处理异常解析器:将处理异常映射到相应的统一错误页面,允许多个实例。
-
视图名称解析器:只允许一个实例。
-
视图解析器:允许多个实例。
注:dispatcherservlet装配的各种组件,有些允许一个实例,有些允许多个实例,如果同一个类型的组件存在多个,可以通过order属性确定优先级。
处理器映射器(handlermapping)和适配器(handleradapter)
handlermapping的作用是将指定的请求url映射为对应的handler(控制器),如果映射成功将返回一个handlerexecutionchain对象(包含一个handler处理器对象,多个handlerinterceptor对象),允许存在多个实例。
handleradapter的作用是将handler包装为适配器进行调用,允许存在多个实例。默认使用dispatcherservlet.properties配置文件中指定的三个实现类分别创建一个适配器。
参数绑定:springmvc将servletrequest对象以及处理方法(请求处理方法)的参数对象实例传递给databinder,databinder会调用装配在springmvc上下文的conversionservice组件进行数据转换、数据格式化转换工作,并将servletrequest中的消息填充到参数对象中。并最终生成数据绑定结果bindingresult对象。bindingresult包含已完成数据绑定的参数对象,还包含相应的校验错误对象。
视图解析器
视图解析器把一个逻辑上的视图名称解析为一个具体的view视图对象,最终的视图可以是jsp、excel、jfreechart等。springmvc返回给用户的视图为具体的view对象,view对象包含model对象。
视图解析流程:
-
springmvc调用目标方法(请求处理方法),将目标方法返回的string、view、modelmap、或modelandview都转换为一个modelandview对象。
-
通过视图解析器将modelandview对象中的view对象进行解析,将逻辑视图view对象解析为一个物理视图view对象。
-
调用物理视图view对象的render()方法进行视图渲染,得到相应结果。
定义多个视图解析器:
所有的viewresolver都实现了ordered接口,在spring中实现了该接口的类都是可以排序的。viewresolver通过order属性来指定顺序,默认都是最大值,可以通过为viewresolver指定order属性来实现viewresolver的优先级,order属性为integer类型,order越小优先级越高。
解析是如果第一个视图解析器返回的view对象为null,则表示该视图解析器不能解析该视图,这时候如果还存在order更大的viewresolver,就会调用剩余的viewresolver中order值最小的那个来解析该视图,以此类推。
注:由于internalresourceviewresolver永远会返回一个非空的view对象,所以一定要把它的order设置为所有存在的viewresolver中值最大的。
更详细的信息可以参考《spring mvc + mybatis快速开发与项目实战》