Spring MVC —— 关于 DispatcherServlet

前言

DispatcherServlet 负责驱动 web 子容器,同时基于此创建对应的 web 组件,本章节结合部分代码理解 DispatcherServlet 的部分细节

DispatcherServlet

DispatcherServlet
整体的继承关系是这样的,因为内容过多,先简单的总结下然后展开:

  • DispatcherServlet 本身是一个 Servlet,继承自 javax.servlet.http.HttpServlet
  • HttpServletBean 复写 init 方法,留出 initServletBean 交给子类拓展
  • 抽象基类 FrameworkServlet 实现 initServletBean 方法驱动 web 子容器
  • FrameworkServlet 同时会被注册为 web 子容器 的监听器,从而回调 onRefreshDispatcherServlet 基于此方法初始化 web 组件
  • FrameworkServlet 收口诸如 doGet doPost 等方法到 processRequest,并提供方法 doService 给子类实现
  • DispatcherServlet 实现 doService,进行一些前置处理后,由 doDispatch 方法基于 web 组件 处理 web 请求

FrameworkServlet#initServletBean

	@Override
	protected final void initServletBean() throws ServletException {
		
		// ...

		try {
			// 初始化 web子容器
			this.webApplicationContext = initWebApplicationContext();

			// 针对 FrameworkServlet 的钩子
			initFrameworkServlet();
		}
		catch (ServletException | RuntimeException ex) {
			throw ex;
		}

		// ...
		
	}

	----------- initWebApplicationContext ----------

	protected WebApplicationContext initWebApplicationContext() {

		// 从 ServletContext 获取根容器
		WebApplicationContext rootContext =
				WebApplicationContextUtils.getWebApplicationContext(getServletContext());
		WebApplicationContext wac = null;

		if (this.webApplicationContext != null) {
			wac = this.webApplicationContext;
			if (wac instanceof ConfigurableWebApplicationContext) {
				ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;
				if (!cwac.isActive()) {
					// 指定根容器为 parent
					if (cwac.getParent() == null) {
						cwac.setParent(rootContext);
					}

					// 配置并启动 web 子容器
					configureAndRefreshWebApplicationContext(cwac);
				}
			}
		}

		// ...

		return wac;
	}
  • DispatcherServletinit 的生命周期阶段,驱动之前(见上章节)创建的 web 子容器
  • 如果存在 根容器,会被指定为 父容器(之所以为什么叫 根容器
  • configureAndRefreshWebApplicationContext 方法对 web 子容器 进行必要的配置并启动(refresh)它,其中包括将 DispatcherServlet 本身注册为 web 子容器ContextRefreshListener

DispatcherServlet#onRefresh

	@Override
	protected void onRefresh(ApplicationContext context) {
		// 组件初始化
		initStrategies(context);
	}

	protected void initStrategies(ApplicationContext context) {
		initMultipartResolver(context);
		initLocaleResolver(context);
		initThemeResolver(context);
		initHandlerMappings(context);
		initHandlerAdapters(context);
		initHandlerExceptionResolvers(context);
		initRequestToViewNameTranslator(context);
		initViewResolvers(context);
		initFlashMapManager(context);
	}
  • DispatcherServlet 本身作为 web 子容器ContextRefreshListener,在容器发布 ContextRefreshedEvent 后,回调 onRefresh 方法
  • onRefresh 方法中,DispatcherServlet 基于启动好的 web 子容器,初始化对应的 web 组件
  • 此处针对组件的初始化代码了解这种模式,对于其中有必要的组件细节会在后续章节学习

initMultipartResolver

	private void initMultipartResolver(ApplicationContext context) {
		try {
			// 从容器中获取 MultipartResolver 实例
			this.multipartResolver = context.getBean(MULTIPART_RESOLVER_BEAN_NAME, MultipartResolver.class);
		}
		// 没有就算了
		catch (NoSuchBeanDefinitionException ex) {
			this.multipartResolver = null;
			if (logger.isTraceEnabled()) {
				logger.trace("...");
			}
		}
	}
  • 这是初始化 MultipartResolver 组件,用来把 HttpServletRequest 转换为 MultipartHttpServletRequest
  • 初始化模式就是从 web 子容器 中获取对应的实例,允许不存在

initLocaleResolver

private void initLocaleResolver(ApplicationContext context) {
		try {
			// 从容器中获取 LocaleResolver 实例
			this.localeResolver = context.getBean(LOCALE_RESOLVER_BEAN_NAME, LocaleResolver.class);
		}
		catch (NoSuchBeanDefinitionException ex) {
			// 如果没有,则基于 DispatcherServlet.properties 配置文件获取默认组件
			this.localeResolver = getDefaultStrategy(context, LocaleResolver.class);
		}
	}
  • 这是初始化 LocaleResolver 组件,用来解析从请求中解析 Locale 以处理 国际化 相关
  • 不同于 MultipartResolver 组件的初始化,如果当前容器中没有对应实例,则会基于 Spring 提供的 DispatcherServlet.properties 来实例化默认组件
  • 默认的 LocaleResolverAcceptHeaderLocaleResolver,即根据 Accept 请求头解析 Locale 实例

小结

  • 基本上上述组件的初始化都是这种模式:从 web 子容器 中获取对应 Bean 实例
  • 如果不存在,对于必须的组件则依据 Spring 的默认配置创建

DispatcherServlet#doDispatch

	protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
		HttpServletRequest processedRequest = request;
		HandlerExecutionChain mappedHandler = null;
		boolean multipartRequestParsed = false;

		WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);

		try {
			ModelAndView mv = null;
			Exception dispatchException = null;

			try {
				/**
				 * 如果有必要,把 HttpServletRequest 转换为 MultipartHttpServletRequest
				 */
				processedRequest = checkMultipart(request);
				multipartRequestParsed = (processedRequest != request);

				// 获取请求对应的 HandlerExecutionChain
				mappedHandler = getHandler(processedRequest);
				if (mappedHandler == null) {
					noHandlerFound(processedRequest, response);
					return;
				}

				// 推断对应的 HandlerAdapter
				HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

				// ...

				/**
				 * 执行所有的 HandlerInterceptor#preHandle
				 * 如果被拦截了就 return
				 */
				if (!mappedHandler.applyPreHandle(processedRequest, response)) {
					return;
				}

				// 请求处理
				mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

				if (asyncManager.isConcurrentHandlingStarted()) {
					return;
				}

				// 视图名称转换
				applyDefaultViewName(processedRequest, mv);

				// 执行所有 HandlerInterceptor#postHandle
				mappedHandler.applyPostHandle(processedRequest, response, mv);
			}
			catch (Exception ex) {
				// ...
			}
			catch (Throwable err) {
				// ...
			}

			// 请求结果的处理
			processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
		}
		
		// ...
		
	}
  • 正如之前所说,DispatcherServlet 作为核心 Servlet,所有的请求处理最终都收口到 doDispatch 方法
  • 粗略地说,这里就是基于之前初始化的 web 组件 来处理客户端的请求,比如 HttpServletRequest 转换、@RequestMapping 的匹配、请求结果的处理 等等
  • 这里是 DispatcherServlet 处理请求的大体流程,旨在了解 DispatcherServlet 的角色和作用,更多的细节实际是委托给对应的 web 组件

总结

本章节针对 DispatcherServlet 的部分功能做了描述:

  • 驱动 web 子容器
  • 基于 web 子容器 初始化 web 组件
  • 基于 web 组件 处理客户端请求
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值