Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
发布日期:2021-06-30 17:55:55 浏览次数:2 分类:技术文章

本文共 7194 字,大约阅读时间需要 23 分钟。

写在前面

在实际项目中,我们更多的是使用Spring的包扫描功能对项目中的包进行扫描,凡是在指定的包或其子包中的类上标注了@Repository、@Service、@Controller、@Component注解的类都会被扫描到,并将这个类注入到Spring容器中。

Spring包扫描功能可以使用XML配置文件进行配置,也可以直接使用@ComponentScan注解进行设置,使用@ComponentScan注解进行设置比使用XML配置文件来配置要简单的多。

使用XML文件配置包扫描

我们可以在Spring的XML配置文件中配置包的扫描,在配置包扫描时,需要在Spring的XML配置文件中的beans节点中引入context标签,如下所示。

接下来,我们就可以在XML配置文件中定义要扫描的包了,如下所示。

整个beans.xml配置文件中的内容如下所示。

这样配置以后,只要在com.meimeixia包下,或者com.meimeixia的子包下标注了@Repository、@Service、@Controller、@Component注解的类都会被扫描到,并自动注入到Spring容器中。

此时,我们分别创建BookDao、BookService以及BookController这三个类,并在这三个类中分别添加@Repository、@Service、@Controller注解,如下所示。

  • BookDao

    package com.meimeixia.dao;import org.springframework.stereotype.Repository;// 名字默认是类名首字母小写@Repositorypublic class BookDao {
    }
  • BookService

    package com.meimeixia.service;import org.springframework.stereotype.Service;@Servicepublic class BookService {
    }
  • BookController

    package com.meimeixia.controller;import org.springframework.stereotype.Controller;@Controllerpublic class BookController {
    }

接下来,我们在工程的src/test/java目录下新建一个单元测试类来进行测试,例如IOCTest。由于我在这儿使用的是junit来进行测试,因此还须在pom文件中添加对junit的依赖,如下所示。

junit
junit
4.12
test

添加完依赖之后,我们就可以在IOCTest测试类中编写如下一个方法来进行测试了,即看一看IOC容器中现在有哪些bean。

package com.meimeixia.test;import org.junit.Test;import org.springframework.context.annotation.AnnotationConfigApplicationContext;import com.meimeixia.config.MainConfig;public class IOCTest {
@SuppressWarnings("resource") @Test public void test() {
ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("beans.xml"); // 我们现在就来看一下IOC容器中有哪些bean,即容器中所有bean定义的名字 String[] definitionNames = applicationContext.getBeanDefinitionNames(); for (String name : definitionNames) {
System.out.println(name); } }}

运行测试用例,输出的结果信息如下图所示。

在这里插入图片描述
可以看到,除了输出我们自己创建的bean的名称之外,也输出了Spring内部使用的一些重要的bean的名称。

接下来,我们使用注解来完成这些功能。

使用注解配置包扫描

使用@ComponentScan注解之前我们先将beans.xml配置文件中的下述配置注释掉。

注释掉之后,我们就可以使用@ComponentScan注解来配置包扫描了。使用@ComponentScan注解配置包扫描非常非常easy!只须在我们的MainConfig类上添加@ComponentScan注解,并将扫描的包指定为com.meimeixia即可,如下所示。

package com.meimeixia.config;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.ComponentScan;import org.springframework.context.annotation.Configuration;import com.meimeixia.bean.Person;/** * 以前配置文件的方式被替换成了配置类,即配置类==配置文件 * @author liayun * */// 这个配置类也是一个组件 @ComponentScan(value="com.meimeixia") // value指定要扫描的包@Configuration // 告诉Spring这是一个配置类public class MainConfig {
// @Bean注解是给IOC容器中注册一个bean,类型自然就是返回值的类型,id默认是用方法名作为id @Bean("person") public Person person01() {
return new Person("liayun", 20); } }

没错,就是这么简单,只需要在类上添加@ComponentScan(value="com.meimeixia")这样一个注解即可。

然后,我们在IOCTest类中新增如下一个test01()方法,以便进行测试。

@SuppressWarnings("resource")@Testpublic void test01() {
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class); // 我们现在就来看一下IOC容器中有哪些bean,即容器中所有bean定义的名字 String[] definitionNames = applicationContext.getBeanDefinitionNames(); for (String name : definitionNames) {
System.out.println(name); }}

运行以上test01()方法,输出的结果信息如下图所示。

在这里插入图片描述
可以看到使用@ComponentScan注解同样输出了容器中bean的名称。

既然使用XML配置文件和注解的方式都能够将相应的类注入到Spring容器当中,那我们是使用XML配置文件还是使用注解呢?我更倾向于使用注解,如果你确实喜欢使用XML配置文件来进行配置,也可以啊,哈哈,个人喜好嘛!好了,我们继续。

关于@ComponentScan注解

我们点开ComponentScan注解类并查看其源码,如下图所示。

在这里插入图片描述
这里,我们着重来看ComponentScan类中的如下两个方法。
在这里插入图片描述
includeFilters()方法指定Spring扫描的时候按照什么规则只需要包含哪些组件,而excludeFilters()方法指定Spring扫描的时候按照什么规则排除哪些组件。两个方法的返回值都是Filter[]数组,在ComponentScan注解类的内部存在Filter注解类,大家可以看下上面的代码。

扫描时排除注解标注的类

现在有这样一个需求,除了@Controller和@Service标注的组件之外,IOC容器中剩下的组件我都要,即相当于是我要排除@Controller和@Service这俩注解标注的组件。要想达到这样一个目的,我们可以在MainConfig类上通过@ComponentScan注解的excludeFilters()方法实现。例如,我们在MainConfig类上添加了如下的注解。

@ComponentScan(value="com.meimeixia", excludeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:除了@Controller和@Service标注的组件之外,IOC容器中剩下的组件我都要,即相当于是我要排除@Controller和@Service这俩注解标注的组件。 */ @Filter(type=FilterType.ANNOTATION, classes={
Controller.class, Service.class})}) // value指定要扫描的包

这样,我们就使得Spring在扫描包的时候排除了使用@Controller和@Service这俩注解标注的类。你要是不信的话,那么可以运行一下IOCTest类中的test01()方法,看看输出的结果信息到底是什么。

在这里插入图片描述
从上图中可以清楚地看到,输出的结果信息中不再输出bookController和bookService了,这已然说明了Spring在进行包扫描时,忽略了@Controller和@Service这俩注解标注的类。

扫描时只包含注解标注的类

我们也可以使用ComponentScan注解类中的includeFilters()方法来指定Spring在进行包扫描时,只包含哪些注解标注的类。

这里需要注意的是,当我们使用includeFilters()方法来指定只包含哪些注解标注的类时,需要禁用掉默认的过滤规则。 还记得我们以前在XML配置文件中配置这个只包含的时候,应该怎么做吗?我们需要在XML配置文件中先配置好use-default-filters="false",也就是禁用掉默认的过滤规则,因为默认的过滤规则就是扫描所有的,只有我们禁用掉默认的过滤规则之后,只包含才能生效。

现在有这样一个需求,我们需要Spring在扫描时,只包含@Controller注解标注的类。要想达到这样一个目的,我们该怎么做呢?可以在MainConfig类上添加@ComponentScan注解,设置只包含@Controller注解标注的类,并禁用掉默认的过滤规则,如下所示。

@ComponentScan(value="com.meimeixia", includeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:我们需要Spring在扫描时,只包含@Controller注解标注的类 */ @Filter(type=FilterType.ANNOTATION, classes={
Controller.class})}, useDefaultFilters=false) // value指定要扫描的包

此时,我们再次运行IOCTest类中的test01()方法,输出的结果信息如下图所示。

在这里插入图片描述
可以看到,在输出的结果中,只包含了@Controller注解标注的组件名称,并没有输出@Service和@Repository这俩注解标注的组件名称。

温馨提示:在使用includeFilters()方法来指定只包含哪些注解标注的类时,结果信息中会一同输出Spring内部的组件名称。

重复注解

不知道小伙伴们有没有注意到ComponentScan注解类上有一个如下所示的注解。

在这里插入图片描述
我们先来看看@ComponentScans注解是个啥,如下图所示。
在这里插入图片描述
可以看到,在ComponentScans注解类的内部只声明了一个返回ComponentScan[]数组的value()方法,说到这里,大家是不是就明白了,没错,这在Java 8中是一个重复注解。

如果你用的是Java 8,那么@ComponentScan注解就是一个重复注解,也就是说我们可以在一个类上重复使用这个注解,如下所示。

@ComponentScan(value="com.meimeixia", includeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:我们需要Spring在扫描时,只包含@Controller注解标注的类 */ @Filter(type=FilterType.ANNOTATION, classes={
Controller.class})}, useDefaultFilters=false) // value指定要扫描的包@ComponentScan(value="com.meimeixia", includeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:我们需要Spring在扫描时,只包含@Service注解标注的类 */ @Filter(type=FilterType.ANNOTATION, classes={
Service.class})}, useDefaultFilters=false) // value指定要扫描的包

运行IOCTest类中的test01()方法,输出的结果信息如下图所示。

在这里插入图片描述
可以看到,同时输出了@Controller注解和@Service注解标注的组件名称。

当然了,如果你使用的是Java 8之前的版本,那也没有问题,虽然我们再也不能直接在类上写多个@ComponentScan注解了,但是我们可以在类上使用@ComponentScans注解,同样也可以指定多个@ComponentScan,如下所示。

@ComponentScans(value={
@ComponentScan(value="com.meimeixia", includeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:我们需要Spring在扫描时,只包含@Controller注解标注的类 */ @Filter(type=FilterType.ANNOTATION, classes={
Controller.class}) }, useDefaultFilters=false), // value指定要扫描的包 @ComponentScan(value="com.meimeixia", includeFilters={
/* * type:指定你要排除的规则,是按照注解进行排除,还是按照给定的类型进行排除,还是按照正则表达式进行排除,等等 * classes:我们需要Spring在扫描时,只包含@Service注解标注的类 */ @Filter(type=FilterType.ANNOTATION, classes={
Service.class}) }, useDefaultFilters=false) // value指定要扫描的包})

再次运行IOCTest类中的test01()方法,输出的结果信息如下图所示。

在这里插入图片描述
与使用多个@ComponentScan注解输出的结果信息相同。

小结

我们可以使用@ComponentScan注解来指定Spring扫描哪些包,可以使用excludeFilters()方法来指定扫描时排除哪些组件,也可以使用includeFilters()方法来指定扫描时只包含哪些组件。当使用includeFilters()方法指定只包含哪些组件时,需要禁用掉默认的过滤规则。

转载地址:https://liayun.blog.csdn.net/article/details/110226283 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:Spring注解驱动开发第4讲——自定义TypeFilter指定@ComponentScan注解的过滤规则
下一篇:Spring注解驱动开发第2讲——使用@Configuration和@Bean给容器中注册组件

发表评论

最新留言

留言是一种美德,欢迎回访!
[***.207.175.100]2024年04月25日 01时46分54秒