当我们用IDE创建好一个SpringMVC项目,会发现集成工具已经帮我们创建了一个类,该类实现了WebApplicationInitializer接口的onStartup方法,就像是main函数一样,整个SpringMVC项目就是从这里开始执行的。那么,这个方法又是在哪里被调用的呢?接下来,我就顺着给捋一捋。
在web容器启动时为提供给第三方组件机会做一些初始化的工作,例如注册servlet或者filtes等,servlet规范中通过ServletContainerInitializer
实现此功能。每个框架要使用ServletContainerInitializer
就必须在对应的jar包的META-INF/services 目录创建一个名为javax.servlet.ServletContainerInitializer
的文件,文件内容指定具体的ServletContainerInitializer
实现类,那么,当web容器启动时就会运行这个初始化器做一些组件内的初始化工作。
一般伴随着ServletContainerInitializer
一起使用的还有HandlesTypes
注解,通过HandlesTypes
可以将感兴趣的一些类注入到ServletContainerInitializer
的onStartup方法作为参数传入。
Tomcat容器的ServletContainerInitializer
机制的实现,主要交由Context容器和ContextConfig监听器共同实现,ContextConfig监听器负责在容器启动时读取每个web应用的WEB-INF/lib
目录下包含的jar包的META-INF/services/javax.servlet.ServletContainerInitializer
,以及web根目录下的META-INF/services/javax.servlet.ServletContainerInitializer
,通过反射完成这些ServletContainerInitializer
的实例化,然后再设置到Context容器中,最后Context容器启动时就会分别调用每个ServletContainerInitializer
的onStartup方法,并将感兴趣的类作为参数传入。
对应到SpringMVC就是:
》ContextConfig
》遍历jar包,找到spring-web-xxx.jar下META-INF/services/javax.servlet.ServletContainerInitializer
文件,找到SpringServletContainerInitializer类。
》通过该类的HandlesTypes
注解@HandlesTypes(WebApplicationInitializer.class),查找项目中所有实现了WebApplicationInitializer接口的类,生成一个集合(上面提到的集成工具为我们创建的类就是实现了WebApplicationInitializer接口)。
》调用SpringServletContainerInitializer的onStartup方法,传入上面的集合和ServletContext对象;
》遍历集合,调用集合元素中的onStartup方法,传入ServletContext对象。
参考:https://blog.csdn.net/wangyangzhizhou/article/details/52013779