为什么80%的码农都做不了架构师?>>>
版本:spring 3.0以上
项目中有两个spring的配置xml,如下
${project.dir}\src\main\resources\META-INF\spring\applicationContext-service-database.xml
${project.dir}\src\main\webapp\WEB-INF\database-servlet.xml
项目中的web.xml
${project.dir}\src\main\webapp\WEB-INF\web.xml
web.xml的配置如下:
contextConfigLocation
classpath*:META-INF/spring/applicationContext-service-database.xml
org.springframework.web.context.ContextLoaderListener
database
org.springframework.web.servlet.DispatcherServlet
1
database
/api/*
1、ContextLoaderListener加载applicationContext-service-database.xml
2、DispatcherServlet加载database-servlet.xml。如果不配置contextConfigLocation的话,DispatcherServlet会默认查找${project.dir}\src\main\webapp\WEB-INF\{servlet-name}-servlet.xml。
如本例:servlet-name=database,则文件名为database-servlet.xml,所有会查找${project.dir}\src\main\webapp\WEB-INF\database-servlet.xml
具体查看文档https://docs.spring.io/spring/docs/3.2.18.RELEASE/spring-framework-reference/htmlsingle/#mvc-introduction中17.2
3、这样配置最终spring会产生两个容器,ContextLoaderListener会生成容器A,DispatcherServlet会生成容器B,容器B的父容器是容器A,容器A的父容器为null
4、容器B会继承容器A的部分功能(其实这样做主要是为了防止单例bean的重复注入和容器B可以获取到容器A的bean,这样进行bean共享,但是容器A获取不到容器B的bean)
5、这样的配置spring mvc就有了,不需要在database-servlet.xml中配置
问题:
如果在applicationContext-service-database.xml配置 ,而在database-servlet.xml没有配置 ,这样就会导致spring mvc功能失效,访问地址会404
原因:
因为如果在applicationContext-service-database.xml配置 ,就会导致ContextLoaderListener加入容器A提前把spring mvc加载到容器A中,但是此时容器A,并没有database-servlet.xml中对应需要向外暴露的接口、url或controller。当DispatcherServlet加载容器B的时候发现,它的父容器A中已经存在spring mvc的相关功能,就不会重复加载,所以就会导致spring mvc的功能失效
关键代码:
org.springframework.web.servlet.DispatcherServlet
private void initHandlerMappings(ApplicationContext context) {
this.handlerMappings = null;
if (this.detectAllHandlerMappings) {
// Find all HandlerMappings in the ApplicationContext, including ancestor contexts.
// BeanFactoryUtils.beansOfTypeIncludingAncestors 会去获取容器B和它的父容器A中的HandlerMapping的接口实现,如果ContextLoaderListener已经配置了
// ,则直接使用ContextLoaderListener加载的HandlerMapping
Map matchingBeans =
BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
if (!matchingBeans.isEmpty()) {
this.handlerMappings = new ArrayList(matchingBeans.values());
// We keep HandlerMappings in sorted order.
OrderComparator.sort(this.handlerMappings);
}
}
else {
try {
HandlerMapping hm = context.getBean(HANDLER_MAPPING_BEAN_NAME, HandlerMapping.class);
this.handlerMappings = Collections.singletonList(hm);
}
catch (NoSuchBeanDefinitionException ex) {
// Ignore, we'll add a default HandlerMapping later.
}
}
// Ensure we have at least one HandlerMapping, by registering
// a default HandlerMapping if no other mappings are found.
// 如果在任何配置文件中都没有配置spring mvc 功能,如: ,则会使用默认的HandlerMapping,默认的HandlerMapping是
//org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping, org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping
// 相关默认配置请查看org\springframework\web\servlet\DispatcherServlet.properties
if (this.handlerMappings == null) {
this.handlerMappings = getDefaultStrategies(context, HandlerMapping.class);
if (logger.isDebugEnabled()) {
logger.debug("No HandlerMappings found in servlet '" + getServletName() + "': using default");
}
}
}
这只是spring mvc 其中一个功能,其他相关功能请查看DispatcherServlet中init*方法