java中MVC原理详解_SpringMVC的工作流程及原理详解

什么是SpringMVC?

Spring MVC是一种基于Java的实现了MVC设计模式的、请求驱动类型的、轻量级Web框架。Spring MVC属于SpringFrameWork的后续产品,已经融合在Spring Web Flow里面。Spring 框架提供了构建 Web 应用程序的全功能 MVC 模块。SpringMVC是一种web层的mvc框架,用于替代servlet(处理响应请求,获取表单参数,表单验证等)。

工作流程图

我个人整理的详细流程图:

88140f367c71fe9f4ac8cf5b3828e1fd.png

工作流程详解

1.由客户端发起请求,到达中央控制器,一般我们配置DispatcherServlet如下:

SpringMVC

org.springframework.web.servlet.DispatcherServlet

contextConfigLocation

/WEB-INF/spring-mvc.xml

1

true

SpringMVC

/

2.中央控制器DispatcherServlet根据请求URI进行解析,找到相应的HandlerMapping(处理器映射器),而HandlerMapping是项目启动就保存在中央控制器中的集合内,我们可以通过源码看到:

3f489a986f0e67422cb717555f00f64e.png

所有的属性都有自己的初始化init方法:

8a22c3c89ccf14d091a211cd0edaa3f0.png

3.DispatcherServlet通过HandlerMapping获取到相应的HandlerExecutionChain(一个执行链对象),发生在getHandler方法内部,未找到就返回空进入另外的操作:

@Nullableprotected HandlerExecutionChain getHandler(HttpServletRequest request) throwsException {if (this.handlerMappings != null) {

Iterator var2= this.handlerMappings.iterator();while(var2.hasNext()) {

HandlerMapping hm=(HandlerMapping)var2.next();if (this.logger.isTraceEnabled()) {this.logger.trace("Testing handler map [" + hm + "] in DispatcherServlet with name '" + this.getServletName() + "'");

}

HandlerExecutionChain handler=hm.getHandler(request);if (handler != null) {returnhandler;

}

}

}return null;

}

4.然后中央控制器DispatcherServlet根据HandlerExecutionChain内的Handler(执行器)对象来匹配相应的HandlerAdapter(处理器适配器),这一步发生在getHandlerAdapter方法内:

protected HandlerAdapter getHandlerAdapter(Object handler) throwsServletException {if (this.handlerAdapters != null) {

Iterator var2= this.handlerAdapters.iterator();while(var2.hasNext()) {

HandlerAdapter ha=(HandlerAdapter)var2.next();if (this.logger.isTraceEnabled()) {this.logger.trace("Testing handler adapter [" + ha + "]");

}if(ha.supports(handler)) {returnha;

}

}

}throw new ServletException("No adapter for handler [" + handler + "]: The DispatcherServlet configuration needs to include a HandlerAdapter that supports this handler");

}

5.适配器(HandlerAdapter)执行handle方法将HandlerExecutionChain内的执行器(Handler)运行(也就是我们写的Controller),并且返回一个ModelAndView模型视图对象:

//mv就是上面定义的ModelAndView对象,mappedHandler就是上面说的HandlerExecutionChain对象

mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

6.中央控制器将接收到的ModelAndView对象进行检查,实际调用的是HandlerInterceptor(处理器拦截器)的postHandle方法:

//注意这个方法是在HandlerExcutionChain类中,调用applyPostHandle检查视图是否存有异常

void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throwsException {

HandlerInterceptor[] interceptors= this.getInterceptors();if (!ObjectUtils.isEmpty(interceptors)) {for(int i = interceptors.length - 1; i >= 0; --i) {

HandlerInterceptor interceptor=interceptors[i];

interceptor.postHandle(request, response,this.handler, mv);

}

}

}

7.上一步没有异常进入视图解析工作,中央控制器将ModelAndView交给视图解析器(View)进行解析(拼接路径,指向视图),并且返回一个真正的View也就是我们的页面。

8.中央控制器接收到View进行数据渲染,填充数据,这两步发生在processDispatchResult方法内:

private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, @Nullable HandlerExecutionChain mappedHandler, @Nullable ModelAndView mv, @Nullable Exception exception) throwsException {boolean errorView = false;if (exception != null) {if (exception instanceofModelAndViewDefiningException) {this.logger.debug("ModelAndViewDefiningException encountered", exception);

mv=((ModelAndViewDefiningException)exception).getModelAndView();

}else{

Object handler= mappedHandler != null ? mappedHandler.getHandler() : null;

mv= this.processHandlerException(request, response, handler, exception);

errorView= mv != null;

}

}if (mv != null && !mv.wasCleared()) {this.render(mv, request, response);if(errorView) {

WebUtils.clearErrorRequestAttributes(request);

}

}else if (this.logger.isDebugEnabled()) {this.logger.debug("Null ModelAndView returned to DispatcherServlet with name '" + this.getServletName() + "': assuming HandlerAdapter completed request handling");

}if (!WebAsyncUtils.getAsyncManager(request).isConcurrentHandlingStarted()) {if (mappedHandler != null) {

mappedHandler.triggerAfterCompletion(request, response, (Exception)null);

}

}

}

9.如果是并发处理,则进行最后的检查,最终响应到客户端,完成一次请求响应。

void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex) throwsException {

HandlerInterceptor[] interceptors= this.getInterceptors();if (!ObjectUtils.isEmpty(interceptors)) {for(int i = this.interceptorIndex; i >= 0; --i) {

HandlerInterceptor interceptor=interceptors[i];try{

interceptor.afterCompletion(request, response,this.handler, ex);

}catch(Throwable var8) {

logger.error("HandlerInterceptor.afterCompletion threw exception", var8);

}

}

}

}

DispatcherServlet主要操作都发生在doDispatch方法内部:

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throwsException {

HttpServletRequest processedRequest= request;//request

HandlerExecutionChain mappedHandler = null;//执行链

boolean multipartRequestParsed = false;//验证是否文件流请求

WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);//异步请求管理器

try{try{

ModelAndView mv= null;//模型视图

Object dispatchException = null;try{

processedRequest= this.checkMultipart(request);//检查是否文件上传相关

multipartRequestParsed = processedRequest != request;//验证请求对象

mappedHandler = this.getHandler(processedRequest);//获取执行链

if (mappedHandler == null) {this.noHandlerFound(processedRequest, response);return;

}

HandlerAdapter ha= this.getHandlerAdapter(mappedHandler.getHandler());//根据执行链中的Handler对象获取对应的适配器

String method =request.getMethod();boolean isGet = "GET".equals(method);if (isGet || "HEAD".equals(method)) {long lastModified =ha.getLastModified(request, mappedHandler.getHandler());if (this.logger.isDebugEnabled()) {this.logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " +lastModified);

}if ((new ServletWebRequest(request, response)).checkNotModified(lastModified) &&isGet) {return;

}

}if (!mappedHandler.applyPreHandle(processedRequest, response)) {return;

}

mv= ha.handle(processedRequest, response, mappedHandler.getHandler());//执行Handler返回模型视图,也就是我们的Controller

if(asyncManager.isConcurrentHandlingStarted()) {return;

}this.applyDefaultViewName(processedRequest, mv);//补充逻辑视图名

mappedHandler.applyPostHandle(processedRequest, response, mv);//执行拦截器的postHandle方法

} catch(Exception var20) {

dispatchException=var20;

}catch(Throwable var21) {

dispatchException= new NestedServletException("Handler dispatch failed", var21);

}this.processDispatchResult(processedRequest, response, mappedHandler, mv, (Exception)dispatchException);//最后渲染工作,最终

} catch(Exception var22) {this.triggerAfterCompletion(processedRequest, response, mappedHandler, var22);

}catch(Throwable var23) {this.triggerAfterCompletion(processedRequest, response, mappedHandler, new NestedServletException("Handler processing failed", var23));

}

}finally{//检查异步请求

if(asyncManager.isConcurrentHandlingStarted()) {if (mappedHandler != null) {

mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);

}

}else if(multipartRequestParsed) {this.cleanupMultipart(processedRequest);

}

}

}

重要组件功能

前端控制器(DispatcherServlet):接收请求,响应结果,相当于转发器,调度其他组件、分发任务,中央处理器

请求到处理器映射(HandlerMapping):根据请求的url查找Handler

处理器适配器(HandlerAdapter):按照特定规则(HandlerAdapter要求的规则)去执行Handler

视图解析器(ViewResolver):进行视图解析,根据逻辑视图名解析成真正的视图(view)

处理器或页面控制器(Handler):执行具体的用户请求,也就是我们的业务Controller

验证器(Validator):验证数据安全

命令对象(command object):请求参数绑定到的对象就叫命令对象

表单对象(form object):请求表单数据对象

SpringMVC特点

清晰的角色划分:控制器(controller)、验证器(validator)、 命令对象(command object)、表单对象(formobject)、模型对象(model object)、 Servlet分发器(DispatcherServlet)、处理器映射(handler mapping)、视图解析器(view resolver)等。每一个角色都可以由一个专门的对象来实现。

强大而直接的配置方式:将框架类和应用程序类都能作为JavaBean配置,支持跨多个context的引用,例如,在web控制器中对业务对象和验证器(validator)的引用。

可适配、非侵入:可以根据不同的应用场景,选择合适的控制器子类 (simple型、command型、form型、wizard型、multi-action型或者自定义),而不是从单一控制器 (比如Action/ActionForm)继承。

可重用的业务代码:可以使用现有的业务对象作为命令或表单对象,而不需要去扩展某个特定框架的基类。

可定制的绑定(binding) 和验证(validation):比如将类型不匹配作为应用级的验证错误, 这可以保存错误的值。再比如本地化的日期和数字绑定等等。在其他某些框架中,你只能使用字符串表单对象,需要手动解析它并转换到业务对象。

可定制的handlermapping和view resolution:Spring提供从最简单的URL映射, 到复杂的、专用的定制策略。与某些webMVC框架强制开发人员使用单一特定技术相比,Spring显得更加灵活。

灵活的model转换:在Springweb框架中,使用基于Map的 键/值对来达到轻易地与各种视图技术的集成。

可定制的本地化和主题(theme)解析:支持在JSP中可选择地使用Spring标签库、支持JSTL、支持Velocity(不需要额外的中间层)等等。

简单而强大的JSP标签库(SpringTag Library):支持包括诸如数据绑定和主题(theme) 之类的许多功能。

JSP表单标签库:在Spring2.0中引入的表单标签库,使得在JSP中编写 表单更加容易。

Spring Bean的生命周期可以被限制在当前的HTTP Request或者HTTP Session。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值