欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则

程序员文章站 2024-03-19 16:29:16
...

写在前面

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

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

使用XML文件配置包扫描

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

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:context="http://www.springframework.org/schema/context" xmlns:p="http://www.springframework.org/schema/p"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://www.springframework.org/schema/beans 
                        http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
                        http://www.springframework.org/schema/context 
                        http://www.springframework.org/schema/context/spring-context-4.2.xsd">

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

<!-- 包扫描:只要是标注了我们熟悉的@Controller、@Service、@Repository、@Component这四个注解中的任何一个的组件,它就会被自动扫描,并加进容器中 -->
<context:component-scan base-package="com.meimeixia"></context:component-scan>

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

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:context="http://www.springframework.org/schema/context" xmlns:p="http://www.springframework.org/schema/p"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://www.springframework.org/schema/beans 
						http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
						http://www.springframework.org/schema/context 
						http://www.springframework.org/schema/context/spring-context-4.2.xsd">
	
	<!-- 包扫描:只要是标注了我们熟悉的@Controller、@Service、@Repository、@Component这四个注解中的任何一个的组件,它就会被自动扫描,并加进容器中 -->
	<context:component-scan base-package="com.meimeixia"></context:component-scan>
	
	<!-- 注册组件 -->
	<bean id="person" class="com.meimeixia.bean.Person">
		<property name="age" value="18"></property>
		<property name="name" value="liayun"></property>
	</bean>
	
</beans>

这样配置以后,只要在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;
    
    // 名字默认是类名首字母小写
    @Repository
    public class BookDao {
        
    }
    
  • BookService

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

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

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

<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
</dependency>

添加完依赖之后,我们就可以在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);
		}
	}

}

运行测试用例,输出的结果信息如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
可以看到,除了输出我们自己创建的Bean的名称之外,也输出了Spring内部使用的一些重要的Bean的名称。

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

使用注解配置包扫描

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

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

注释掉之后,我们就可以使用@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")
@Test
public void test01() {
    AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
    // 我们现在就来看一下IOC容器中有哪些Bean,即容器中所有Bean定义的名字
    String[] definitionNames = applicationContext.getBeanDefinitionNames();
    for (String name : definitionNames) {
        System.out.println(name);
    }
}

运行以上test01()方法,输出的结果信息如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
可以看到使用@ComponentScan注解同样输出了容器中Bean的名称。

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

关于@ComponentScan注解

我们点开ComponentScan注解类并查看其源码,如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
这里,我们着重来看ComponentScan类中的如下两个方法。
Spring注解驱动开发第3讲——使用@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()方法,看看输出的结果信息到底是什么。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
从上图中可以清楚地看到,输出的结果信息中不再输出bookController和bookService了,这已然说明了Spring在进行包扫描时,忽略了@Controller和@Service这俩注解标注的类。

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

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

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

<context:component-scan base-package="com.meimeixia" use-default-filters="false"></context:component-scan>

现在有这样一个需求,我们需要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()方法,输出的结果信息如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
可以看到,在输出的结果中,只包含了@Controller注解标注的组件名称,并没有输出@Service和@Repository这俩注解标注的组件名称。

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

重复注解

不知道小伙伴们有没有注意到ComponentScan注解类上有一个如下所示的注解。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
我们先来看看@ComponentScans注解是个啥,如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
可以看到,在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()方法,输出的结果信息如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
可以看到,同时输出了@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()方法,输出的结果信息如下图所示。
Spring注解驱动开发第3讲——使用@ComponentScan自动扫描组件并指定扫描规则
与使用多个@ComponentScan注解输出的结果信息相同。

小结

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