SpringMVC的执行流程

一、SpringMVC的执行流程

  1. 客户端发送请求到这个DispatcherServlet前端控制器
  2. 前端控制器会通过这个HandlerMapping处理器映射器,找到合适的处理器,就是通过这个输入的url,找到对应的handler
  3. 返回处理器的执行链,里面会包含多个拦截器的信息,以及需要查找的处理器handler的信息
  4. 找处理器适配器HandlerAdapter,这一步开始就会去调用handler里面的方法
  5. 通过执行这个handler里面的方法,会去找具体的controller方法
  6. 找到具体的controller之后,会返回一个modelAndView给这个HanderAdapter给处理器适配器
  7. 处理器适配器获取到ModelAndView之后,会将这个结果返回给DispatcherServlet前端控制器
  8. 通过这个ViewResolver视图解析器进行解析这个ModelAndView
  9. 解析完成之后,会将这个view返回给前端DispatcherServlet前端控制器
  10. 将model中的数据填充到这个view视图里面,最后去渲染视图

二、源码分析

1,首先打开这个DispatcherServlet类,从源码可以发现这个类是继承了FrameworkServlet

public class  DispatcherServlet  extends  FrameworkServlet{ ...}

2,这个类里面有一个doService方法,里面有一个比较重要的方法,叫doDispatch方法

doDispatch(request, response);

3,进入这个**doDispatch()**方法里面,可以看到以下几行代码,主要是来返回这个处理器执行链,处理器适配器等操作。

//处理器执行链
HandlerExecutionChain mappedHandler = null;
//检测当前请求是否需要做文件上传
processedRequest = checkMultipart(request);
//获取需要的映射器以及拦截器等
mappedHandler = getHandler(processedRequest);
if (mappedHandler == null) { 
	noHandlerFound(processedRequest, response);
	return;
}
//找这个处理器适配器
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
//调用拦截器 
mappedHandler.applyPreHandle(processedRequest, response);
//适配器开始调用这个handler
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
//处理这个结果集
processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);

4,接下来所有的分析都基于这个**doDispatch()**方法,来对里面的方法做一个具体的描述。

接下来看看这个getHandler方法,主要是通过这个request请求,来获取对应handler。

protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception { 
	if (this.handlerMappings != null) { 
        //遍历全部获取到的处理器映射器
		for (HandlerMapping hm : this.handlerMappings) { 
            //通过这个请求进行匹配,找到这个handler
			HandlerExecutionChain handler = hm.getHandler(request);
            //找到则返回
			if (handler != null) { 
				return handler;
			}
        }
	}
	return null;
}

5,接下来看看这个获取处理器适配器getHandlerAdapter的方法,适配器也有多种,会根据不同的handler适配不同的适配器。

protected HandlerAdapter getHandlerAdapter(Object handler) throws ServletException { 
	if (this.handlerAdapters != null) { 
		for (HandlerAdapter ha : this.handlerAdapters) { 
			if (ha.supports(handler)) { 
				return ha;
			}
		}
	}
	throw new ServletException();
}

6,再看这个doDispatch方法中的handle方法,可以点进去发现具体的实现如下。最终会以一个controller的对象返回

public  ModelAndView  handle(HttpServletRequest request,  HttpServletResponse response,  Object handler)  throws  Exception  { 
	//以一个controller的方式返回 
	return ((Controller) handler).handleRequest(request, response);
}

7,可以自定义重写这个handleRequest方法,最后以一个ModelAndView的格式返回给这个处理适配器,处理器适配器将这个ModelAndView返回给前端控制器。

@Override
public  ModelAndView  handleRequest(HttpServletRequest request,  HttpServletResponse response)  throws  Exception  { 
	ModelAndView mv = new ModelAndView();
	mv.setViewName("success");
	mv.addObject("hello","zhs");
	return mv;
}

8,接下来看看这个前端控制器处理这个ModelAndView的视图解析,主要是通过这个processDispatchResult方法来实现。

if (mv != null && !mv.wasCleared()) { 
    //如果这个modelAndView不为空,就会开始进入正式的解析
	render(mv, request, response);
	if (errorView) { 
		WebUtils.clearErrorRequestAttributes(request);
	}
}

再进入这个render方法里面,可以发现这个视图解析器会对这个视图进行一个具体的解析,并且会将解析的view返回给这个DispatcherServlet里面,最终会将这个数据进行一个转发或者重定向,对这个view进行一个渲染,最终响应到这个客户端。

//会进行一个具体的解析
view = resolveViewName(viewName, mv.getModelInternal(), locale, request);
//解析
view.render(mv.getModelInternal(), request, response);
//输出
renderMergedOutputModel(mergedModel, getRequestToExpose(request), response);
//对这个获取的数据进行转发或者重定向
RequestDispatcher rd = getRequestDispatcher(request, dispatcherPath);
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值