博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
项目启动之spring篇
阅读量:6593 次
发布时间:2019-06-24

本文共 4214 字,大约阅读时间需要 14 分钟。

在之前终于是把项目跑起来了,也是踩了不少的坑,把bug还原了一遍又一遍,希望能看的更清楚。现在我们跳过tomcat再来看看spring是如何启动的。

                   项目启动过程之spring篇

1.创建WebApplicationContext

# 首先我们在web.xml的配置了这么一个lisenner

<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

# 从源码中找到该类,类中的方法都是调用父类来实现的,如此明显的装饰者模式。

public class ContextLoaderListener extends ContextLoader implements ServletContextListener {

    .....

}

# 再看看ContextLoader这个类

public class ContextLoader {

   # 这个静态方法只为加载一个ContextLoader.properties文件,不出所料的这个文件跟类在同一个文件夹下。

   # 文件中只有一句话 -> org.springframework.web.context.WebApplicationContext=org.springframework.web.context.support.XmlWebApplicationContext

   static {

        try {

            ClassPathResource resource = new ClassPathResource("ContextLoader.properties", ContextLoader.class);

            defaultStrategies = PropertiesLoaderUtils.loadProperties(resource);

        } catch (IOException var1) {

            throw new IllegalStateException("Could not load 'ContextLoader.properties': " + var1.getMessage());

        }

        currentContextPerThread = new ConcurrentHashMap(1);

    }

    # 这个方法大致就是如果web.xml没有配置特殊的contextClass则会使用defaultStrategies中指定的类,也就是上面静态方法加载到的配置 XmlWebApplicationContext

    protected Class<?> determineContextClass(ServletContext servletContext) {

        String contextClassName = servletContext.getInitParameter("contextClass");

        if (contextClassName != null) {

            try {

                return ClassUtils.forName(contextClassName, ClassUtils.getDefaultClassLoader());

            } catch (ClassNotFoundException var4) {

                throw new ApplicationContextException("Failed to load custom context class [" + contextClassName + "]", var4);

            }

        } else {

            # 如果在web.xml中并未指定contextClass

            # 简单写成: contextClassName = "org.springframework.web.context.support.XmlWebApplicationContext"

            contextClassName = defaultStrategies.getProperty(WebApplicationContext.class.getName());

            try {

                return ClassUtils.forName(contextClassName, ContextLoader.class.getClassLoader());

            } catch (ClassNotFoundException var5) {

                throw new ApplicationContextException("Failed to load default context class [" + contextClassName + "]", var5);

            }

        }

    }

    protected WebApplicationContext createWebApplicationContext(ServletContext sc) {

       # 将里面代码简单写成:

       # return (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(this.determineContextClass(sc));

    }

    public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {

        if (servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {

           ....

        } else {

            .....

            # 此时就要开始创建 WebApplicationContext 了

            try {

                if (this.context == null) {

                    # 我们假定 this.context 的类型就是 XmlWebApplicationContext

                    this.context = this.createWebApplicationContext(servletContext);

                }

                if (this.context instanceof ConfigurableWebApplicationContext) {

                    ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext)this.context;

                    

                    if (!cwac.isActive()) {   # 刚跑起的项目context一般都没被激活

                        if (cwac.getParent() == null) {

                            # 很奇怪,loadParentContext返回的是  return null  ??

                            ApplicationContext parent = this.loadParentContext(servletContext);

                            cwac.setParent(parent);

                        }

                        # 这个方法主要是设置contextId,若没有在web.xml中指定,则会像这个样子:

                        #   org.springframework.web.context.WebApplicationContext:${contextPath}

                        this.configureAndRefreshWebApplicationContext(cwac, servletContext);

                    }

                }

                # 这个才是Lisener的关键,把创建的WebApplicationContext加入到servletContext中。 

                servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);

                ClassLoader ccl = Thread.currentThread().getContextClassLoader();

                if (ccl == ContextLoader.class.getClassLoader()) {

                    currentContext = this.context;

                } else if (ccl != null) {

                    currentContextPerThread.put(ccl, this.context);

                }

                return this.context;

            .....

        }

    }

}

# 在XWAC中提供了默认的配置文件位置

public class XmlWebApplicationContext extends AbstractRefreshableWebApplicationContext {

    public static final String DEFAULT_CONFIG_LOCATION = "/WEB-INF/applicationContext.xml";

    public static final String DEFAULT_CONFIG_LOCATION_PREFIX = "/WEB-INF/";

    public static final String DEFAULT_CONFIG_LOCATION_SUFFIX = ".xml";

    .....

}

小结一下:   tomcat启动时会给配置在web.xml中指定的lisener传递一个事件,让其初始化创建一个WebApplicationContext对象,并将其加入到ServletContext中。而在spring-web中ContextLoaderListener类为我们提供使用默认或自定义两种生成WebApplicationContext对象,需要使用自定义则需要在web.xml中指定contextClass属性,并且创建对应的类。创建好了context之后,下一步便是加载applicationContext中的bean了,暂时也还没有发现如何发起加载bean的过程,继续学习吧。。。

转载地址:http://mgcio.baihongyu.com/

你可能感兴趣的文章
线程池以及四种常见线程池
查看>>
网络编程-第一节
查看>>
基于JAVA的银行卡实名认证接口调用代码实例
查看>>
Centos下安装并设置nginx开机自启动
查看>>
bzoj3195: [Jxoi2012]奇怪的道路
查看>>
C# 使用 CancellationTokenSource 终止线程
查看>>
idea创建springboot工程
查看>>
Web中间件漏洞
查看>>
简单轮播图
查看>>
微信公众号页面无法唤起输入框
查看>>
day 32并行 并发
查看>>
Mac上安装stf
查看>>
介绍几个移动web app开发框架
查看>>
十六进制转十进制(蓝桥杯)
查看>>
搭建Easyui环境在Myeclipse或Eclipse中
查看>>
bin log、redo log、undo log和MVVC
查看>>
ubuntu 重启网络方法--通过杀死进程重启网络
查看>>
深度优先搜索(DFS)(Java)
查看>>
Java --Serializable序列化
查看>>
He angrily answer MBT Tunisha
查看>>