目录

无注解方式component-scan使用

注解方式@ComponentScan使用

@ComponentScan的扫描规则


无注解方式component-scan使用

之前,我们需要扫描工程下一些类上所标注的注解,这些常用注解有:

@Controller,@Service,@Component,@Repository

通过在Spring的配置文件中配置<context:component-scan>扫描对应包下扫描这些注解的方式:

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:p="http://www.springframework.org/schema/p"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context  http://www.springframework.org/schema/context/spring-context-2.5.xsd"><!--@Controller,@Service,@Component,@Repository--><context:component-scan base-package="com.jektong.spring"/></beans>

注解方式@ComponentScan使用

建三个类,依次将

@Controller,@Repository,@Service,标注这些类:

图1

现在通过使用注解@ComponentScan的方式来扫描所在包下面的这些类:之前定义的PersonConfig修改:

package com.jektong.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;import com.jektong.spring.Person;@Configuration
@ComponentScan("com.jektong")
public class PersonConfig {@Bean("person01")public Person person() {return new Person("李四",21);}
}

测试,看是否扫描到这些注解所标注的类:PersonTest.java

@Test
public  void test02() {ApplicationContext ac = new AnnotationConfigApplicationContext(PersonConfig.class);Person bean = ac.getBean(Person.class);System.out.println(bean);String[] beanDefinitionNames = ac.getBeanDefinitionNames();for (String string : beanDefinitionNames) {System.out.println(string);}
}

测试效果:除了Spring要自动加载的配置类以外也显示了刚才添加的配置类:

图2

为何会出现PersonConfig,因为@Configuration本 身就是@Component注解的:

图3

@ComponentScan的扫描规则

如果需要指定配置类的扫描规则, @ComponentScan提供对应的扫描方式@Filter进行配置类的过滤:

// 扫描包的时候只规定扫描一些注解配置类。
Filter[] includeFilters() default {};// 扫描包的时候可以排除一些注解配置类。
Filter[] excludeFilters() default {};

Filter其实也是一个注解,相当于@ComponentScan的子注解,可以看图4

图4

Filter对应的过滤规则如下:

第一种:扫描包的时候只规定扫描一些注解配置类【includeFilters】。

使用这个includeFilters过滤规则,必须解除默认的过滤规则,

使用【useDefaultFilters = false】:

package com.jektong.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;import com.jektong.spring.Person;@Configuration
@ComponentScan(value = "com.jektong",includeFilters  = {@Filter(type = FilterType.ANNOTATION,value= {Controller.class})
},useDefaultFilters = false )
public class PersonConfig {@Bean("person01")public Person person() {return new Person("李四",21);}
}

这样就只会扫描用@Controller,标注的配置类交给Spring容器中了:

图5

第二种:扫描包的时候可以排除一些注解配置类【excludeFilters 】。

图6

@Filter看上图,有5种不同类型的过滤策略。拿第一种举例,我们需要过滤使用@Controller注解的配置类:

package com.jektong.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;import com.jektong.spring.Person;@Configuration
@ComponentScan(value = "com.jektong",excludeFilters = {@Filter(type = FilterType.ANNOTATION,value= {Controller.class})
} )
public class PersonConfig {@Bean("person01")public Person person() {return new Person("李四",21);}
}

测试看一下发现图2中的personController不会交给Spring容器去管理了:

图7

上面的图6展示出5种不同类型的过滤策略,上面介绍了注解类型(FilterType.ANNOTATION),还有四种:

重点看一下CUSTOM自定义扫描策略。

从源码看,自定义扫描注解类型需要实现TypeFilter接口,下面就写一个实现类MyFilter.java:在实现类中可以自定义配置规则:

package com.jektong.config;import java.io.IOException;import org.springframework.core.io.Resource;
import org.springframework.core.type.AnnotationMetadata;
import org.springframework.core.type.ClassMetadata;
import org.springframework.core.type.classreading.MetadataReader;
import org.springframework.core.type.classreading.MetadataReaderFactory;
import org.springframework.core.type.filter.TypeFilter;public class MyFilter implements TypeFilter {@Overridepublic boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory)throws IOException {// 查看当前类的注解。AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata();// 查看当前扫描类的信息ClassMetadata classMetadata = metadataReader.getClassMetadata();// 获取当前类资源Resource resource = metadataReader.getResource();String className = classMetadata.getClassName();System.out.println("className===>" + className);// 只要类名包含er则注册Spring容器if(className.contains("er")) {return true;}return false;}
}

测试:

PersonConfig 中进行扫描:

package com.jektong.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;import com.jektong.service.PersonService;
import com.jektong.spring.Person;@Configuration
@ComponentScan(value = "com.jektong",includeFilters  = {@Filter(type = FilterType.CUSTOM,value= {MyFilter.class})
},useDefaultFilters = false )
public class PersonConfig {@Bean("person01")public Person person() {return new Person("李四",21);}
}

可以看出扫描出包下面的类只要带“er”的全部扫描出来,并配置给Spring容器:

ASSIGNABLE_TYPE:按照指定的类型去加载对应配置类:

package com.jektong.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;import com.jektong.service.PersonService;
import com.jektong.spring.Person;@Configuration
@ComponentScan(value = "com.jektong",includeFilters  = {@Filter(type = FilterType.ASSIGNABLE_TYPE,value= {PersonService.class})
},useDefaultFilters = false )
public class PersonConfig {@Bean("person01")public Person person() {return new Person("李四",21);}
}

尽管我们将PersonService.java上的注解去掉,使用ASSIGNABLE_TYPE依然会加载出来(自行测试)。

ASPECTJ与REGEX基本不用,不用了解。

以上就是@ComponentScan的具体用法,该兴趣的话可以看一下源码。

Spring注解详解:@ComponentScan自动扫描组件使用相关推荐

  1. spring注解详解与用法(总览)

    这篇文章收集了我写的所有的spring注解的详细说明与用法,点击可以跳转到对应文章,此文章会不断更新 spring注解详解与用法(1)最基础也是最常见的如下所示,详情点击这里 @Controller/ ...

  2. Spring注解详解包扫描bean注册

    一. @Configuration 配置 ​ 告诉Spring容器这是一个配置类 ==xml配置 二. @ComponentScan 包扫描 ​ 说明: //value:指定要扫描的包 //按照规则指 ...

  3. spring注解详解与用法(3)切面环绕

    导航:更多的spring注解标签点击这里 声明了切面类我们还需要定义切面的内容面向切面编程是spring的一个重要思想,也是精华之一 @Aspect 这是一个切面的注解,声明是一个额外的面向切面的功能 ...

  4. Spring注解详解

    概述 注释配置相对于 XML 配置具有很多的优势: 它可以充分利用 Java 的反射机制获取类结构信息,这些信息可以有效减少配置的工作.如使用 JPA 注释配置 ORM 映射时,我们就不需要指定 PO ...

  5. Spring 注解详解

    使用注解来构造IoC容器 在spring 3.0中,通过@controller标注即可将class定义为一个controller类.为使spring能找到定义为controller的bean,需要在a ...

  6. Spring实现自动装配(spring注解详解)和手动注入比较

    概述 注释配置相对于 XML 配置具有很多的优势: 它可以充分利用 Java 的反射机制获取类结构信息,这些信息可以有效减少配置的工作.如使用 JPA 注释配置 ORM 映射时,我们就不需要指定 PO ...

  7. @Configuration Spring注解详解

    第一个作用 使用注释类@Configuration表示它的主要目的是作为bean定义的来源 第二个作用 @Configuration类允许通过简单地调用@Bean同一类中的其他方法来定义bean间依赖 ...

  8. spring注解详解,看完你就都全懂了!!

    转载:https://blog.csdn.net/walkerjong/article/details/7946109

  9. Spring零配置之@Configuration注解详解

    转载自 Spring零配置之@Configuration注解详解 @Configuration介绍 Spring3.0之前要使用Spring必须要有一个xml配置文件,这也是Spring的核心文件,而 ...

最新文章

  1. 控制器框架Struts与策略模式那点事
  2. 8、collection
  3. Android SlidingMenu以及ActionBarSherlock的基础使用教程
  4. 重置linux桌面,Ubuntu 18.04小贴士:重置Gnome桌面与使用隐藏的屏幕录像工具
  5. P3349-[ZJOI2016]小星星【树形dp,容斥】
  6. django构建网页_如何使用Django构建照片供稿
  7. AttributeError: module 'pymysql' has no attribute 'escape' 错误的出现以及解决
  8. rost反剽窃检测系统_动静结合,最硬核反外挂
  9. 设定pic单片机端口为输入_PIC单片机入门_输入输出端口详解
  10. 基于Java+Swing+Mysql项目信息管理系统
  11. viewpro.php是啥意思,海信 Hi-View Pro系列 画质引擎芯片简介
  12. Atomic类如何保证原子性
  13. php mysql 聊天室_聊天室phpmysql(二)_php
  14. linux下pip升级
  15. SpringBoot+Vue实现前后端分离的学校快递站点管理系统
  16. 已知信码序列为1011_专升本计算机网络:校验码
  17. EBS 报表开发:XML Publisher Excel模板
  18. 使用NFC模拟校园卡门禁功能 【Mac, Windows, Android, 手环】
  19. BLDC 电机的反电动势电压仿真
  20. python实现简易聊天需要登录_python socket编程 实现简单p2p聊天程序

热门文章

  1. 国密sm4js整理,js 调用,易语言调用
  2. 白色简洁大方公司企业网站源码 WordPress主题2款
  3. 【线性代数】多项式各类算法总结
  4. 数字电子技术基础(八):超前进位加法器
  5. 2021建筑施工电工(建筑特种作业)机考题库及建筑电工模拟试题
  6. 浅谈宽带卫星通信系统
  7. 模糊数学Python(二)模糊聚类分析
  8. Unix时间戳与系统毫秒的转换
  9. CRM系统促进企业高效管理
  10. 对于运维来说,可靠性到底是个啥?