亲宝软件园·资讯

展开

Spring MVC 处理请求 Spring MVC 处理一个请求的流程

程序员自由之路 人气:0
想了解Spring MVC 处理一个请求的流程的相关内容吗,程序员自由之路在本文为您仔细讲解Spring MVC 处理请求的相关知识和一些Code实例,欢迎阅读和指正,我们先划重点:Spring,MVC,处理请求,Spring,MVC,处理请求的流程,下面大家一起来学习吧。

一个请求从客户端发出到达服务器,然后被处理的整个过程其实是非常复杂的。本博客主要介绍请求到达服务器被核心组件DispatcherServlet处理的整理流程(不包括Filter的处理流程)。

1. 处理流程分析

Servlet处理一个请求时会调用service()方法,所以DispatcherServlet处理请求的方式也是从service()方法开始(debug的话建议从DispatcherServlet的service方法开始debug)。FrameworkServlet重写了HttpServlet的service方法,这个service方法后面又调用了FrameworkServlet的processRequest()方法,processRequest()调用了DispatcherServlet的doService()方法,最后调用到DispatcherServlet的doDispatcher()方法。整合处理请求的方法调用流程如上,下面看下代码:

protected void service(HttpServletRequest request, HttpServletResponse response)
		throws ServletException, IOException {

	HttpMethod httpMethod = HttpMethod.resolve(request.getMethod());
	if (HttpMethod.PATCH == httpMethod || httpMethod == null) {
		processRequest(request, response);
	}
	else {
 //这边调用了HttpServlet的service()方法,但由于FrameWorkServle重写了doGet、doPost等方法,所以最终还是会调用到processRequest方法
		super.service(request, response);
	}
}

再看看FrameworkServlet的processRequest()方法。

 protected final void processRequest(HttpServletRequest request, HttpServletResponse response)
 		throws ServletException, IOException {
 
 	long startTime = System.currentTimeMillis();
 	Throwable failureCause = null;
 
 	LocaleContext previousLocaleContext = LocaleContextHolder.getLocaleContext();
 	LocaleContext localeContext = buildLocaleContext(request);
 
 	RequestAttributes previousAttributes = RequestContextHolder.getRequestAttributes();
 	ServletRequestAttributes requestAttributes = buildRequestAttributes(request, response, previousAttributes);
 
 	WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
 	asyncManager.registerCallableInterceptor(FrameworkServlet.class.getName(), new RequestBindingInterceptor());
 
 	initContextHolders(request, localeContext, requestAttributes);
 
 	try {
  //这边调用DispatcherServlet的doService()方法
 		doService(request, response);
 	}
 	catch (ServletException ex) {
 		failureCause = ex;
 		throw ex;
 	}
 	catch (IOException ex) {
 		failureCause = ex;
 		throw ex;
 	}
 	catch (Throwable ex) {
 		failureCause = ex;
 		throw new NestedServletException("Request processing failed", ex);
 	}
 
 	finally {
 		resetContextHolders(request, previousLocaleContext, previousAttributes);
 		if (requestAttributes != null) {
 			requestAttributes.requestCompleted();
 		}
 
 		if (logger.isDebugEnabled()) {
 			if (failureCause != null) {
 				this.logger.debug("Could not complete request", failureCause);
 			}
 			else {
 				if (asyncManager.isConcurrentHandlingStarted()) {
 					logger.debug("Leaving response open for concurrent processing");
 				}
 				else {
 					this.logger.debug("Successfully completed request");
 				}
 			}
 		}
 
 		publishRequestHandledEvent(request, response, startTime, failureCause);
 	}
 }

doService()方法的具体内容会在后面讲到,这边描述下doDispatcher()的内容,

首先根据请求的路径找到HandlerMethod(带有Method反射属性,也就是对应Controller中的方法),然后匹配路径对应的拦截器,有了HandlerMethod和拦截器构造个HandlerExecutionChain对象。HandlerExecutionChain对象的获取是通过HandlerMapping接口提供的方法中得到。有了HandlerExecutionChain之后,通过HandlerAdapter对象进行处理得到ModelAndView对象,HandlerMethod内部handle的时候,使用各种HandlerMethodArgumentResolver实现类处理HandlerMethod的参数,使用各种HandlerMethodReturnValueHandler实现类处理返回值。 最终返回值被处理成ModelAndView对象,这期间发生的异常会被HandlerExceptionResolver接口实现类进行处理。

总结下Spring MVC处理一个请求的过程:

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返回响应给用户,到此一个流程结束。

2. 请求流程图

还是这个图比较清楚。发现根据代码不太能把这个流程说清楚。而且整个流程很长,代码很多,我就不贴代码了。这里根据这个图再把整个流程中组件的功能总结下:

上面的HandlerMethod和HandlerInterceptor组件分别对应我们Controller中的方法和拦截器。拦截器会在HandlerMethod方法执行之前执行

当HandlerAdapter组件执行完成之后会得到一个ModleAndView组件,这个组件代表视图模型。

加载全部内容

相关教程
猜你喜欢
用户评论