• Spring注解详解:@ComponentScan自动扫描组件使用


    目录

    无注解方式component-scan使用

    注解方式@ComponentScan使用

    @ComponentScan的扫描规则


    无注解方式component-scan使用

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

    @Controller,@Service,@Component,@Repository

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

    
    
        
    	
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    注解方式**@ComponentScan**使用

    建三个类,依次将

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

    9676e1d230404d359e879afc90694c96.png

    图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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    测试,看是否扫描到这些注解所标注的类: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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

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

    32af07a5fb0f442c916164f7f441bd35.png

    图2

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

    f3647d616e5145e68ca4f5c2c8a21616.png

    图3

    @ComponentScan的扫描规则

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

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

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

    0616dc7cd25a4dcdb5ac759f50ab4230.png

    图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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

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

    692b89ffdc8b4038977d5b4b8ebbbf94.png

    图5

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

    87f83f3d10824d1cb846cea234642191.png

    图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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

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

    1157d6b18b2f40d89f59310b3c624bf4.png

    图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 {
    
    	@Override
    	public 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;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31

    测试:

    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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    可以看出扫描出包下面的类只要带“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);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

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

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

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

  • 相关阅读:
    【每日随笔】驾驭人性 ② ( 员工立场问题 | 立场转变 | 吴越同舟 | 老板如何与员工结成利益共同体 )
    【数字IC基础】DFT(Design For Test)可测性设计
    为什么低代码CRM越来越受欢迎?
    手把手教你使用 Spring Boot 3 开发上线一个前后端分离的生产级系统(一) - 介绍
    自学(黑客技术)——网络安全高效学习方法
    Ubuntu - 安装 MySQL 8
    DAO的精简化治理的委托机制
    个人开发者轻松接入支付回调
    价格监测千万别漏掉这些内容
    Unity & PS Linear Workflow - Unity 和 PS 的线性工作流实践 - 简单配置示例(后续补上渲染差异图)
  • 原文地址:https://blog.csdn.net/m0_67392182/article/details/126041789