一、背景

​ 我们在Spring+SpringMVC+Mybatis的集成开发中,经常会遇到事务配置不起作用等问题,有时候就是因为包扫描出了问题,其中component-scan的标签的use-default-filters属性坑了很多人,那么本文就来分析下出现这种问题可能的原因以及解决方式。

二、分析及原理窥探

项目结构

我们在spring-mvc.xml文件中进行如下配置,这种方式会成功扫描到带有@Controller注解的Bean,不会扫描带有@Service@Repository注解的Bean,是正确的。

<context:component-scan base-package="com.hafiz.www.controller">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

但是如下方式配置不仅仅扫描到带有@Controller注解的Bean,还扫描到带有@Service@Repository注解的Bean,可能造成事务不起作用等问题。

<context:component-scan base-package="com.hafiz.www"></context:component-scan>

这是因为什么呢?下面让我们来从源码进行分析:

  1. <context:component-scan/>会交给org.springframework.context.config.ContextNamespaceHandler处理.

    registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser());
  2. ComponentScanBeanDefinitionParser会读取配置文件信息并组装成org.springframework.context.annotation.ClassPathBeanDefinitionScanner进行处理。
  3. 如果没有配置<context:component-scan>use-default-filters属性,则默认为true,在创建ClassPathBeanDefinitionScanner时会根据use-default-filters是否为true来调用如下代码:
    protected void registerDefaultFilters() {
      this.includeFilters.add(new AnnotationTypeFilter(Component.class));
      ClassLoader cl = ClassPathScanningCandidateComponentProvider.class.getClassLoader();
      try {
        this.includeFilters.add(new AnnotationTypeFilter(
          ((Class<? extends Annotation>) cl.loadClass("javax.annotation.ManagedBean")), false));
        logger.info("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning");
      }
      catch (ClassNotFoundException ex) {
        // JSR-250 1.1 API (as included in Java EE 6) not available - simply skip.
      }
      try {
        this.includeFilters.add(new AnnotationTypeFilter(
          ((Class<? extends Annotation>) cl.loadClass("javax.inject.Named")), false));
        logger.info("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");
      }
      catch (ClassNotFoundException ex) {
        // JSR-330 API not available - simply skip.
      }
    }

    从以上源码我们可以看出默认ClassPathBeanDefinitionScanner会自动注册对@Component@ManagedBean@Named注解的Bean进行扫描。

  4. 在进行扫描时会通过include-filter/exclude-filter来判断你的Bean类是否是合法的:
    protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException {
    for (TypeFilter tf : this.excludeFilters) {
    if (tf.match(metadataReader, this.metadataReaderFactory)) {
    return false;
    }
    }
    for (TypeFilter tf : this.includeFilters) {
    if (tf.match(metadataReader, this.metadataReaderFactory)) {
    AnnotationMetadata metadata = metadataReader.getAnnotationMetadata();
    if (!metadata.isAnnotated(Profile.class.getName())) {
    return true;
    }
    AnnotationAttributes profile = MetadataUtils.attributesFor(metadata, Profile.class);
    return this.environment.acceptsProfiles(profile.getStringArray("value"));
    }
    }
    return false;
    }

    从以上源码可看出:扫描时首先通过exclude-filter 进行黑名单过滤,然后通过include-filter 进行白名单过滤,否则默认排除。

    三、结论

    在spring-mvc.xml中进行如下配置:

    <context:component-scan base-package="com.hafiz.www">
    <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

    则SpringMVC容器不仅仅扫描并注册带有@Controller注解的Bean,而且还扫描并注册了带有@Component的子注解@Service@Reposity的Bean。因为use-default-filters默认为true。所以如果不需要默认的,则use-default-filters=“false”禁用掉。叉车价格
    当我们进行上面的配置时,SpringMVC容器会把service、dao层的bean重新加载,从而造成新加载的bean覆盖了老的bean,但事务的AOP代理没有配置在spring-mvc.xml配置文件中,造成事务失效。解决办法是:在spring-mvc.xml配置文件中的context:component-scan标签中使用use-default-filters=“false”禁用掉默认的行为。

最新文章

  1. occ添加新的捕捉模式
  2. JavaScript 对象的创建和对6种继承模式的理解和遐想
  3. 解决Python2.7的UnicodeEncodeError: &#39;ascii&#39; codec can’t encode异常错误
  4. Daily Scrum 10.31
  5. matlab中实现Gabor滤波器
  6. Jenkins 搭建U3D自动发布 IOS
  7. flex_播放视频_本地_与_FMS端
  8. HTML5 push
  9. Google生活
  10. CSS样式中字体乱码
  11. Hibernat之关系的处理一对一处理
  12. db2 调整连接数的优化
  13. Java注解学习笔记
  14. String放入运行时常量池的时机与String.intern()方法解惑
  15. linux IPtable防火墙 禁止和开放端口
  16. Redis配置解读
  17. python中range()函数的用法
  18. 纯CSS绘制mac代码
  19. 让zepto支持slideup(),slidedown()
  20. Android launchMode SingleTask newIntent 的问题

热门文章

  1. #leetcode刷题之路27-移除元素
  2. Ubuntu下配置jdk和tomcat
  3. 什么是ajax和json,说说他们的优缺点
  4. Sppring MVC核心应用-2
  5. Flume的介绍和简单操作
  6. 项目-高性能TcpServer - 目录
  7. 基于STM32的简易数码相册
  8. keil5 配置 stm32f103rc 软件仿真
  9. Java设计模式(24)——行为模式之解释器模式(Interpreter)
  10. 20155220 实验三 敏捷开发与XP实践 实验报告