源码地址

springboot2教程系列

在spring-boot框架中@EnableAutoConfiguration 开启自动扫描装配Bean

实现方式

  • 注解方式
  • 编程方式

背景

在学习SpringBoot的时候,我们会使用到@Enable***注解的地方,使用上也都是加在@Configuration 类注解的类上面,比如:
(1)@EnableAutoConfiguration 开启自动扫描装配Bean

(2)@EnableScheduling 开启计划任务的支持

(3)@EnableTransactionManagement 开启注解式事务的支持。

(4)@EnableCaching开启注解式的缓存支持。

(5)@EnableAspectJAutoProxy 开启对AspectJ自动代理的支持,

(6) @EnableAsync 开启异步方法的支持

(7) @EnableWebMvc 开启Web MVC的配置支持。

(8) @EnableConfigurationProperties 开启对@ConfigurationProperties注解配置Bean的支持。

(9)@EnableJpaRepositories 开启对Spring Data JPA Repository的支持。

查看@Enable的源码

查看这些注解的实现,我们发现每一个注解都有一个 @Import 注解。@Import注解在4.2之前只支持导入配置类,在4.2,@Import注解支持导入普通的java类,并将其声明成一个bean。这也说明了,自动开启的实现,其实是导入了一些配置类。

@EnableWebMvc采用注解驱动方式

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(DelegatingWebMvcConfiguration.class)
public @interface EnableWebMvc {
}

@EnableCaching采用接口编程方式

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({CachingConfigurationSelector.class})
public @interface EnableCaching {
    boolean proxyTargetClass() default false;

    AdviceMode mode() default AdviceMode.PROXY;

    int order() default 2147483647;
}


public class CachingConfigurationSelector extends AdviceModeImportSelector<EnableCaching> {

   ......
   
    public CachingConfigurationSelector() {
    }

    public String[] selectImports(AdviceMode adviceMode) {
        switch(adviceMode) {
        case PROXY:
            return this.getProxyImports();
        case ASPECTJ:
            return this.getAspectJImports();
        default:
            return null;
        }
    }

   .....

}
  • 首先@EnableCaching需要@Import一个CachingConfigurationSelector类
  • CachingConfigurationSelector又继承AdviceModeImportSelector
  • AdviceModeImportSelector又实现了ImportSelector
  • 所有,当我们自定以一个@Enable模块时需要ImportSelector的实现类

自定义@Enable模块

基于注解驱动实现

定义给一个配置类

@Configuration
public class MyConfiguration {
    /**
     * 方法名为 Bean 名称
     * @return
     */
    @Bean
    public String test() {
        return "自定义配置类";
    }
}

使用注解的方式,@Import需要导入一个配置类。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({MyConfiguration.class})
public @interface EnableMyConfig {
}
@SpringBootApplication
@EnableMyConfig
public class TestApplication {

    public static void main(String[] args){
        SpringApplication app = new SpringApplication(TestApplication.class);
        // app.setWebEnvironment(false);
        ConfigurableApplicationContext context =app.run(args);

        String test = context.getBean("test",String.class);
        System.out.println("bean是否存在--->"+test);
    }
}

测试时TestApplication类跟MyConfiguration类要放在不同包下,或者不使用@SpringBootApplication,使用@Configuration+@EnableAutoConfiguration。

@SpringBootApplication=@Configuration+@EnableAutoConfiguration+@ComponentScan,其中扫描包的范围为启动类所在包和子包,不包括第三方的jar包。如果我们需要扫描通过maven依赖添加的jar,我们就要单独使用@ComponentScan注解扫描第三方包。

基于接口驱动实现

配置类

public class MyConfiguration2 {
    /**
     * 方法名为 Bean 名称
     * @return
     */
    @Bean
    public String test2() {
        return "自定义配置类";
    }
}

自定义ImportSelector的实现类。

public class MyImportSelector2 implements ImportSelector {

    @Override
    public String[] selectImports(AnnotationMetadata annotationMetadata) {
        // 返回一个自定义类
        // 使用MyImportSelector导入MyConfiguration,而不是直接导入MyConfiguration
        return new String[]{MyConfiguration2.class.getName()};
    }
}

自定义@Enable

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({MyImportSelector2.class})
public @interface EnableMyConfig2 {
}

测试

@Configuration
@EnableAutoConfiguration
@EnableMyConfig2
public class EnableApplication {
	
	public static void main(String[] args){
        SpringApplication app = new SpringApplication(EnableApplication.class);
       // app.setWebEnvironment(false);
        ConfigurableApplicationContext context =app.run(args);

        String test = context.getBean("test2",String.class);
        System.out.println("bean是否存在--->"+test);
    }

}
  • 运行顺序
    首先,EnableTestBootstrap启动,之后根据@EnableMyConfig注解,找到MyImportSelector类,找到selectImports方法,进行加载“test”bean操作。
  • 对比
    这种实现方式,我们可以在装载bean中间进行一些操作,相对注解实现方式比较自由,编程实现方式可以有一些弹性操作。