上一篇笔记(Spring MVC源码——Root WebApplicationContext)中记录了下 Root WebApplicationContext 的初始化代码.这一篇来看 Servlet WebApplicationContext 的初始化代码html
DispatcherServlet
是另外一个须要在 web.xml
中配置的类, Servlet WebApplicationContext 就由它来建立和初始化.git
HttpServletBean
简单继承了 HttpServlet
, 负责将 init-param 中的参数注入到当前 Servlet
实例的属性中, 而且为子类提供了增长 requiredProperties 的能力. HttpServletBean
并不依赖于 Spring 容器.github
来看一下它的 init()
方法:web
public final void init() throws ServletException { // Set bean properties from init parameters. // 从 ServletConfig 中取出初始化参数到 PropertyValues。ServletConfigPropertyValues 的构造器中将会检查是否缺失了必要属性 PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties); if (!pvs.isEmpty()) { try { // 将 servlet 对象包装成 BeanWrapper ,从而可以以 Spring 的方式(反射)来注入参数 BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this); ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext()); // 注册 PropertyEditor,遇到 Resource 类型的属性时,用 ResourceEditor 解析 bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment())); // 初始化 BeanWrapper,空方法 initBeanWrapper(bw); // 注入属性,忽略没有 setter 的属性 bw.setPropertyValues(pvs, true); } catch (BeansException ex) { if (logger.isErrorEnabled()) { logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex); } throw ex; } } // Let subclasses do whatever initialization they like. // 由子类实现初始化逻辑 initServletBean(); }
private static class ServletConfigPropertyValues extends MutablePropertyValues { public ServletConfigPropertyValues(ServletConfig config, Set<String> requiredProperties) throws ServletException { // 将 requiredProperties 拷贝到新的 Set missingProps Set<String> missingProps = (!CollectionUtils.isEmpty(requiredProperties) ? new HashSet<>(requiredProperties) : null); // 将 ServletConfig 中的初始化参数取出,添加到 MutablePropertyValues 中 Enumeration<String> paramNames = config.getInitParameterNames(); while (paramNames.hasMoreElements()) { String property = paramNames.nextElement(); Object value = config.getInitParameter(property); addPropertyValue(new PropertyValue(property, value)); if (missingProps != null) { missingProps.remove(property); } } // Fail if we are still missing properties. if (!CollectionUtils.isEmpty(missingProps)) { // 存在必须出现的条件没出现 throw new ServletException( "Initialization from ServletConfig for servlet '" + config.getServletName() + "' failed; the following required properties were missing: " + StringUtils.collectionToDelimitedString(missingProps, ", ")); } } }
FrameworkServlet
是一个更具体的 Servlet 基类. 它有如下两个功能:spring
WebApplicationContext
实例.FrameworkServlet
重写了 HttpServletBean
的 initServletBean()
方法, 这个方法会在 全部 servlet 的属性被注入以后执行, 来看一下代码:springboot
protected final void initServletBean() throws ServletException { getServletContext().log("Initializing Spring " + getClass().getSimpleName() + " '" + getServletName() + "'"); if (logger.isInfoEnabled()) { logger.info("Initializing Servlet '" + getServletName() + "'"); } long startTime = System.currentTimeMillis(); try { // 初始化 webApplicationContext this.webApplicationContext = initWebApplicationContext(); // 在容器被加载后执行,由子类来实现一些必要的初始化 initFrameworkServlet(); } catch (ServletException | RuntimeException ex) { logger.error("Context initialization failed", ex); throw ex; } // 略去打印日志的部分 ...
}
initWebApplicationContext()
方法会初始化并返回一个容器:mvc
protected WebApplicationContext initWebApplicationContext() { // 获取 Root WebApplicationContext WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(getServletContext()); WebApplicationContext wac = null; if (this.webApplicationContext != null) { // A context instance was injected at construction time -> use it // 一个上下文已经被注入进来 wac = this.webApplicationContext; if (wac instanceof ConfigurableWebApplicationContext) { // 若是是 ConfigurableWebApplicationContext, ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac; if (!cwac.isActive()) { // 没有激活,设置父容器,配置而且刷新容器 if (cwac.getParent() == null) { cwac.setParent(rootContext); } configureAndRefreshWebApplicationContext(cwac); } } } if (wac == null) { // 尝试从 ServletContext 中获取一个容器 wac = findWebApplicationContext(); } if (wac == null) { // 建立一个新的容器并初始化 wac = createWebApplicationContext(rootContext); } if (!this.refreshEventReceived) { // 没有触发过刷新时间 synchronized (this.onRefreshMonitor) { // 手动触发刷新事件 onRefresh(wac); } } if (this.publishContext) { // Publish the context as a servlet context attribute. // 将容器发布到 ServletContext 的属性上 String attrName = getServletContextAttributeName(); getServletContext().setAttribute(attrName, wac); } return wac; }
onRefresh()
方法供子类来重写, DispatcherServlet
重写了这个方法来初始化 MVC 中的一些组件:app
@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); }
initWebApplicationContext()
方法调用的其余方法其实和 ContextLoader
中的方法比较相似, 这里就再也不放上来了, 有兴趣的能够访问个人源码注释.webapp
经过本篇博客以及上一篇博客,相信你们对springmvc的上下文有了明确的认识。总的来讲,默认springmvc项目会有两个上下文(root webapplicationcontext 和 servlet webapplicationcontext)。接下来的博文会带你们认识一下springboot项目的上下文以及springmvc项目上下文和springboot项目上下文的异同点,敬请期待。ide
转自:https://www.cnblogs.com/FJH1994/p/10813687.html