揭密springboot自动装配(2)--AutoConfigurationImportSelector

 

我们从启动类里的@SpringBootApplication注解开始


里面有三个主要注解

@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan

这里自动装配的主要注解是@EnableAutoConfiguration这个,我们跟进去看看

会发现这里面有引入我们上章提到的神秘东东AutoConfigurationImportSelector

继续跟进AutoConfigurationImportSelector一探究竟,它是实现DeferredImportSelector,这个里面继承了ImportSelector

我们来看看他的selectImports方法实现

@Override
  public String[] selectImports(AnnotationMetadata annotationMetadata) {
    if (!isEnabled(annotationMetadata)) {
      return NO_IMPORTS;
    }
    AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader
        .loadMetadata(this.beanClassLoader);
    AutoConfigurationEntry autoConfigurationEntry = getAutoConfigurationEntry(autoConfigurationMetadata,
        annotationMetadata);
    return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
  }

这里面我们关注getAutoConfigurationEntry继续跟进去

protected AutoConfigurationEntry getAutoConfigurationEntry(AutoConfigurationMetadata autoConfigurationMetadata,
      AnnotationMetadata annotationMetadata) {
    if (!isEnabled(annotationMetadata)) {
      return EMPTY_ENTRY;
    }
    AnnotationAttributes attributes = getAttributes(annotationMetadata);
    List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
    configurations = removeDuplicates(configurations);
    Set<String> exclusions = getExclusions(annotationMetadata, attributes);
    checkExcludedClasses(configurations, exclusions);
    configurations.removeAll(exclusions);
    configurations = filter(configurations, autoConfigurationMetadata);
    fireAutoConfigurationImportEvents(configurations, exclusions);
    return new AutoConfigurationEntry(configurations, exclusions);
  }

继续跟进getCandidateConfigurations这个方法

  protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
    List<String> configurations = SpringFactoriesLoader.loadFactoryNames(getSpringFactoriesLoaderFactoryClass(),
        getBeanClassLoader());
    Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you "
        + "are using a custom packaging, make sure that file is correct.");
    return configurations;
  }

到这里你会发现有这么句话No auto configuration classes found in META-INF/spring.factories,

META-INF/spring.factories这个文件是不是有点熟悉,先放着,继续跟进就会明白了

public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
        String factoryTypeName = factoryType.getName();
        return (List)loadSpringFactories(classLoader).getOrDefault(factoryTypeName, Collections.emptyList());
    }
private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
        MultiValueMap<String, String> result = (MultiValueMap)cache.get(classLoader);
        if (result != null) {
            return result;
        } else {
            try {
                Enumeration<URL> urls = classLoader != null ? classLoader.getResources("META-INF/spring.factories") : ClassLoader.getSystemResources("META-INF/spring.factories");
                LinkedMultiValueMap result = new LinkedMultiValueMap();

                while(urls.hasMoreElements()) {
                    URL url = (URL)urls.nextElement();
                    UrlResource resource = new UrlResource(url);
                    Properties properties = PropertiesLoaderUtils.loadProperties(resource);
                    Iterator var6 = properties.entrySet().iterator();

                    while(var6.hasNext()) {
                        Entry<?, ?> entry = (Entry)var6.next();
                        String factoryTypeName = ((String)entry.getKey()).trim();
                        String[] var9 = StringUtils.commaDelimitedListToStringArray((String)entry.getValue());
                        int var10 = var9.length;

                        for(int var11 = 0; var11 < var10; ++var11) {
                            String factoryImplementationName = var9[var11];
                            result.add(factoryTypeName, factoryImplementationName.trim());
                        }
                    }
                }

                cache.put(classLoader, result);
                return result;
            } catch (IOException var13) {
                throw new IllegalArgumentException("Unable to load factories from location [META-INF/spring.factories]", var13);
            }
        }
    }

直到这里你会发现,他就是加载了所有META-INF/spring.factories文件下的数据,我们随便点开一个META-INF/spring.factories看看

这里面就有很多需要自动装配的key-value对,到这里我们就应该能猜到大概实现了

这里loadFactoryNames就是获取对应EnableAutoConfiguration.class也就是对应上面spring.factories文件里的key为org.springframework.boot.autoconfigure.EnableAutoConfiguration的所有value值也就是class路径

protected Class<?> getSpringFactoriesLoaderFactoryClass() {
		return EnableAutoConfiguration.class;
	}

我们重新回到selectImports方法

到这里我们就很清楚这里返回的就是所有在spring.factories文件里key为org.springframework.boot.autoconfigure.EnableAutoConfiguration的所有value值class路径,印证上一章中的例子,我们下面回顾下:

同样这里是不是可以启发我们,将配置类写在META-INF/spring.factories文件里,然后key写为

org.springframework.boot.autoconfigure.EnableAutoConfiguration

是不是也可以自动帮我们给装配进去,例如下面这样我们在自己项目中创建META-INF/spring.factories,我们自动装配UserA

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.example.demo.service.UserA

这样我们也是一样可以在spring容器中可以拿到我们的UserA;

ok,基本的实现流程我们已经了解大概了,那到这里应该还是会有个疑问,哪个地方调用selectImports方法?

下章我们就这个问题进行分析

 

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
### 回答1: Spring Boot自动装配是指Spring Boot根据一定的规则,自动为我们的应用程序配置依赖关系。这种方式可以大大简化我们的开发工作,因为我们不需要手动配置所有的bean,而只需要添加所需的依赖关系即可。 Spring Boot自动装配通过条件注解实现,即在满足一定条件下自动装配所需的bean。例如,当我们在应用程序中添加了spring-boot-starter-web依赖时,Spring Boot会自动为我们配置Tomcat服务器和Spring MVC框架。 Spring Boot还提供了许多其他的starter依赖,如spring-boot-starter-data-jpa、spring-boot-starter-data-redis等,它们都会自动配置所需的bean,我们只需要添加相应的依赖即可。 当然,如果我们需要自定义配置,也可以通过@Configuration注解来实现。这样我们就可以覆盖自动配置的默认值,实现自己的需求。 ### 回答2: Spring Boot自动装配是指Spring Boot能够根据特定的条件自动配置应用程序所需的各种组件和依赖项。它通过使用注解和条件判断来完成自动配置。 Spring Boot自动装配具有以下几个特点: 1. 条件注解:Spring Boot会根据应用程序中的条件注解来判断是否要自动装配某个组件。例如,@ConditionalOnClass注解表示只有当指定的类存在于类路径上时,才会进行自动装配。 2. 自动配置类:Spring Boot提供了大量的自动配置类,这些类中包含了一些常用组件的配置信息,如数据源、缓存、Web服务器等。开发人员只需引入相应的依赖,Spring Boot就会自动根据这些依赖来配置应用程序。 3. 自定义配置:除了使用Spring Boot提供的自动配置类外,开发人员还可以创建自己的自动配置类,以满足特定的需求。自定义配置类需要使用@Configuration注解,并将所需的组件添加到Spring容器中。 4. 自动装配的优先级:当存在多个自动配置类时,Spring Boot会根据装配的优先级来确定使用哪个配置类。开发人员可以通过@AutoConfigureOrder注解或实现Ordered接口来定义自动配置的优先级。 总的来说,Spring Boot自动装配能够简化应用程序的开发和部署过程,开发人员无需手动配置大量的组件和依赖项,只需引入相关的依赖,Spring Boot就能完成相应的配置工作。这大大提高了开发效率,使得开发人员能够更专注于核心业务逻辑的实现。 ### 回答3: SpringBoot自动装配是指通过框架提供的机制,自动完成项目中的各种组件的依赖注入和配置工作。SpringBoot借助自动装配可以大大简化项目的配置和开发工作。 首先,SpringBoot通过注解和配置文件的方式实现自动装配。使用注解可以轻松地标识出哪些组件需要被自动装配,并指定装配的方式。配置文件则提供了更加灵活的配置方式,可以对组件的各种属性进行详细的配置。 其次,SpringBoot自动装配是基于条件的。即只有满足一定的条件时,才会进行自动装配。条件可以是类的存在、配置属性的值、环境变量等等。这样可以根据项目的实际需求,只装配需要的组件,避免了无用的组件的装配和加载,提高了项目的性能和效率。 另外,SpringBoot提供了很多内置的自动装配组件和功能。例如,可以自动装配数据源、Web容器、数据库连接池等等。这些功能都是通过简单的配置和注解就可以实现,无需开发者自己去手动配置和加载。 总而言之,SpringBoot自动装配的好处在于简化了项目的配置和开发工作,提高了开发效率和代码的可维护性。通过合理地利用自动装配机制,能够快速搭建项目框架,减少重复劳动,并且保证了组件的正确装配和配置。这也是SpringBoot广受开发者喜爱的原因之一。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值