天天看點

Spring注解——使用@ComponentScan自動掃描元件

文章系轉載,友善整理和檢視

作者:雲原生實戰

連結:https://www.jianshu.com/p/64aac6461d5b

來源:簡書

1.建立一個配置類,在配置類上添加 @ComponentScan 注解。該注解預設會掃描該類所在的包下所有的配置類,相當于之前的 context:component-scan。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;

@ComponentScan
public class BeanConfig {

}
           

2.使用 ApplicationContext 的 getBeanDefinitionNames() 方法擷取已經注冊到容器中的 bean 的名稱。

import io.mieux.config.BeanConfig;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class App02 {
    public static void main(String[] args) {
        ApplicationContext applicationContext =
                new AnnotationConfigApplicationContext(BeanConfig.class);

        String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames();
        for (String beanName : beanDefinitionNames) {
            System.out.println("beanName: " + beanName);
        }
    }
}
           

運作效果:

beanName: org.springframework.context.annotation.internalConfigurationAnnotationProcessor
beanName: org.springframework.context.annotation.internalAutowiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalRequiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalCommonAnnotationProcessor
beanName: org.springframework.context.event.internalEventListenerProcessor
beanName: org.springframework.context.event.internalEventListenerFactory
beanName: beanConfig
           

除了 spring 本身注冊的一些 bean 之外,可以看到最後一行,已經将 BeanConfig 這個類注冊進容器中了。

3.指定要掃描的包(使用@ComponentScan 的 valule 屬性來配置),建立一個controller 包,并在該包下建立一個 AppController 類。

package io.mieux.controller;

import org.springframework.stereotype.Controller;

@Controller
public class AppController {
}
           

在類上加了@Controller注解,說明該類是一個 Component。在 BeanConfig 類中修改:

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;

@ComponentScan(value = "io.mieux.controller")
public class BeanConfig {

}
           

在 @ComponentScan 注解中指定了要掃描的包。

運作效果:

beanName: org.springframework.context.annotation.internalConfigurationAnnotationProcessor
beanName: org.springframework.context.annotation.internalAutowiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalRequiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalCommonAnnotationProcessor
beanName: org.springframework.context.event.internalEventListenerProcessor
beanName: org.springframework.context.event.internalEventListenerFactory
beanName: beanConfig
beanName: appController
           

AppController 已經被注冊進容器了。

4.excludeFilters 和 includeFilters 的使用

使用 excludeFilters 來按照規則排除某些包的掃描。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;

@ComponentScan(value = "io.mieux",
        excludeFilters = {@Filter(type = FilterType.ANNOTATION,
        value = {Controller.class})})
public class BeanConfig {

}
           

excludeFilters 的參數是一個 Filter[] 數組,然後指定 FilterType 的類型為 ANNOTATION,也就是通過注解來過濾,最後的 value 則是Controller 注解類。配置之後,在 spring 掃描的時候,就會跳過 io.mieux 包下,所有被 @Controller 注解标注的類。

使用 includeFilters 來按照規則隻包含某些包的掃描。

在建立一個 service 的包,并建立一個 AppService 類,再加上一個 @Service 注解。

package io.mieux.service;

import org.springframework.stereotype.Service;

@Service
public class AppService {
}
           

修改 BeanCofig 類:

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;

@ComponentScan(value = "io.mieux", includeFilters = {@Filter(type = FilterType.ANNOTATION, classes = {Controller.class})})
public class BeanConfig {

}
           

運作效果:

beanName: org.springframework.context.annotation.internalConfigurationAnnotationProcessor
beanName: org.springframework.context.annotation.internalAutowiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalRequiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalCommonAnnotationProcessor
beanName: org.springframework.context.event.internalEventListenerProcessor
beanName: org.springframework.context.event.internalEventListenerFactory
beanName: beanConfig
beanName: appController
beanName: appService
           

配置裡面,應該是隻包含 @Controller 注解的類才會被注冊到容器中,為什麼 @Service 注解的類也被注冊了呢?這裡涉及到 @ComponentScan 的一個 useDefaultFilters 屬性的用法,該屬性預設值為 true,也就是說 spring 預設會自動發現被 @Component、@Repository、@Service 和 @Controller 标注的類,并注冊進容器中。要達到隻包含某些包的掃描效果,就必須将這個預設行為給禁用掉(在 @ComponentScan 中将 useDefaultFilters 設為 false 即可)。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;

@ComponentScan(value = "io.mieux", 
        includeFilters = {@Filter(type = FilterType.ANNOTATION, classes = {Controller.class})},
        useDefaultFilters = false)
public class BeanConfig {

}
           

運作效果:

beanName: org.springframework.context.annotation.internalConfigurationAnnotationProcessor
beanName: org.springframework.context.annotation.internalAutowiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalRequiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalCommonAnnotationProcessor
beanName: org.springframework.context.event.internalEventListenerProcessor
beanName: org.springframework.context.event.internalEventListenerFactory
beanName: beanConfig
beanName: appController
           

5.添加多種掃描規則

1、如果使用的 jdk8,則可以直接添加多個 @ComponentScan 來添加多個掃描規則,但是在配置類中要加上 @Configuration 注解,否則無效。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

@ComponentScan(value = "io.mieux.controller")
@ComponentScan(value = "io.mieux.service")
@Configuration
public class BeanConfig {

}
           

2、也可以使用 @ComponentScans 來添加多個 @ComponentScan,進而實作添加多個掃描規則。同樣,也需要加上 @Configuration 注解,否則無效。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScans;
import org.springframework.context.annotation.Configuration;

@ComponentScans(value = 
        {@ComponentScan(value = "io.mieux.controller"),
        @ComponentScan(value = "io.mieux.service")})
@Configuration
public class BeanConfig {

}
           

6.添加自定義過濾規則

在前面使用過 @Filter 注解,裡面的 type 屬性是一個 FilterType 的枚舉類型:

public enum FilterType {

    ANNOTATION,
    ASSIGNABLE_TYPE,
    ASPECTJ,
    REGEX,
    CUSTOM

}
           

使用 CUSTOM 類型,就可以實作自定義過濾規則。

1、 首先建立一個實作 TypeFilter 接口的 CustomTypeFilter 類,并實作其 match 方法。

package io.mieux.config;

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;

import java.io.IOException;

public class CustomTypeFilter implements TypeFilter {
    @Override
    public boolean match(MetadataReader metadataReader,
                         MetadataReaderFactory metadataReaderFactory) throws IOException {

        // 擷取目前掃描到的類的注解中繼資料
        AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata();
        // 擷取目前掃描到的類的中繼資料
        ClassMetadata classMetadata = metadataReader.getClassMetadata();
        // 擷取目前掃描到的類的資源資訊
        Resource resource = metadataReader.getResource();

        if (classMetadata.getClassName().contains("Co")) {
            return true;
        }

        return false;
    }
}
           

這裡簡單對掃描到的類名進行判斷,如果類名包含”Co“的就符合條件,也就會注入到容器中。

2、對 BeanConfig 進行修改,指定過濾類型為 Custom 類型,并指定 value 為 CustomTypeFilter.class。

package io.mieux.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.FilterType;

@ComponentScan(value = "io.mieux",
        includeFilters = {@ComponentScan.Filter(type = FilterType.CUSTOM, value = {CustomTypeFilter.class})},
        useDefaultFilters = false)
public class BeanConfig {

}
           

運作效果:

beanName: org.springframework.context.annotation.internalConfigurationAnnotationProcessor
beanName: org.springframework.context.annotation.internalAutowiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalRequiredAnnotationProcessor
beanName: org.springframework.context.annotation.internalCommonAnnotationProcessor
beanName: org.springframework.context.event.internalEventListenerProcessor
beanName: org.springframework.context.event.internalEventListenerFactory
beanName: beanConfig
beanName: appController