Spring component-scan 的逻辑 、单例模式下多实例问题、事务失效

原创内容,转发请保留:http://www.cnblogs.com/iceJava/p/6930118.html,非常欢迎指正,谢谢

之前遇到该问题,今天查看了下 spring 4.x 的代码

一,先理解下 context:component-scan 处理过程:
 1 <!-- scan the package and the sub package -->
 2 <!--
 3        【重要】:容易产生事务失效的地方,见:http://jinnianshilongnian.iteye.com/blog/1762632
 4        处理逻辑(入口见:org.springframework.context.config.ContextNamespaceHandler):
 5        1. context:component-scan 表示是否扫描指定路径下的所有 .class 文件;
 6            参见 org.springframework.core.io.support.PathMatchingResourcePatternResolver.getResources 方法
 7 
 8        2. use-default-filters,是否使用将 默认的 AnnotationTypeFilter 增加到 include-filter 过滤器,包括:
 9            @Component(包括子注解@Service、@Reposity、@Controller)、@ManagedBean、@Named注解 的Bean
10            也就是说,如果使用 use-default-filters = true, 就不需要 <context:annotation-config/> 配置了
11            参见 org.springframework.context.annotation.ComponentScanBeanDefinitionParser.configureScanner
12 
13 
14        3. 遍历所有的bean, 保留满足 "不是 exclude-filter 黑名单、并且至少满足一条 include-filter 白名单" .class 文件
15            参见 org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider.isCandidateComponent 方法    
16 --> 17 <context:component-scan base-package="com.hm.mobile" use-default-filters="false" > 18 <context:include-filter type="regex" expression=".*Controller$" /> 19 <context:include-filter type="regex" expression=".*Interceptor$" /> 20 </context:component-scan>

二:单例模式下的多实例问题

1. 包结构:

  

MessageController、MessageConsumerListener、MessageProviderService 分别 使用了 @Controller @Service @Service

 

2. application.xml 配置文件片段

 

1 <context:component-scan base-package="com.hm.mobile" >
2     <context:exclude-filter type="regex" expression=".*Controller$" />
3     <context:exclude-filter type="regex" expression=".*Interceptor$" />
4 </context:component-scan>

 

  不难理解,在该配置下,Spring 容器 扫描到:MessageConsumerListener、MessageProviderService 并实例化, 其中 MessageController 被黑名单排除

3. dispatcher-servlet.xml 配置文件片段

1 <context:component-scan base-package="com.hm.mobile">
2     <context:include-filter type="regex" expression=".*Controller$" />
3     <context:include-filter type="regex" expression=".*Interceptor$" />
4 </context:component-scan>

  Spring MVC 容器 扫描到了 MessageConsumerListener、MessageProviderService、MessageController 并实例化 

 

4. 通过 vituralVM,我们得到了证实:

 

5. 在 application-servlet.xml 增加  use-default-filters="false"  之后

 


三:事务失效问题

  a. 一般情况下,我们都是在 Spring 配置中事务,因此只有 spring 容器中的 service 对象 被 AOP 注入事务;换句话说, Spring MVC 容器中的service对象 是没有进行 AOP 事务注入的

     b. 因为 Spring mvc 中注入的 service 对象是 自身容器中的 对象,而不是 Spring 容器的

     备注:第三点,是本人自己的猜想,没来得及去验证。

原文地址:https://www.cnblogs.com/iceJava/p/6930118.html