脚本宝典收集整理的这篇文章主要介绍了SpringBoot自动装配,脚本宝典觉得挺不错的,现在分享给大家,也给大家做个参考。
一. SPRing注解发展过程
SpringBoot的自动装配依赖于注解,所以我们先来看一下注解的发展过程。
以下主要对核心注解进行说明
public class MyImportSelector implements ImportSelector {//继承该接口 @override //重写selectImports方法 public String[] selectImports(AnnotationMetadata importingClassMetadata) { //返回对象对应的类型的全类路径的字符串数组 return new String[]{XXX1.class.getName(), XXX2.class.getName()}; } }
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {//继承该接口 @Override //重写registerBeanDefinitions方法 public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { // 将需要注册的对象封装为 RootBeanDefinition 对象 RootBeanDefinition xxx1 = new RootBeanDefinition(XXX1.class); registry.registerBeanDefinition("xxx1", xxx1); //再注册一个 RootBeanDefinition xxx2 = new RootBeanDefinition(XXX2.class); registry.registerBeanDefinition("xxx2", xxx2); } }
/** * 定义一个 Condition 接口的是实现 */ public class MyCondition implements Condition { @Override public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) { //业务逻辑... return false; // 默认返回false }}
//使用 @Configuration public class JavaConfig { @Bean // 条件注解,添加的类型必须是 实现了 Condition 接口的类型 // MyCondition的 matches 方法返回true 则注入,返回false 则不注入 @Conditional(MyCondition.class) public StudentService studentService() { return new StudentService(); } }
二. SpringBoot自动装配原理
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan
@Import(AutoConfigurationImportSelector.class)
它的内部主要是使用@import注解导入一个选择器。
3.那么我们看看这个AutoConfigurationImportSelector类
上文提到继承ImportSelector接口的类,需要重写 selectImports( ),那我们就看看这个方法
@Override public String[] selectImports(AnnotationMetadata annotationMetadata) { if (!isEnabled(annotationMetadata)) { return NO_IMPORTS; } AutoConfigurationEntry autoConfigurationEntry = getAutoConfigurationEntry(annotationMetadata); return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations()); }
该方法其实也没说啥,现在的重心就放在getAutoConfigurationEntry()中
4.getAutoConfigurationEntry()
protected AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) { if (!isEnabled(annotationMetadata)) { return EMPTY_ENTRY; } AnnotationAttributes attributes = getAttributes(annotationMetadata); //获取候选配置信息,加载的是当前项目的classpath目录下的所有的 spring.factories 文件中的 key 为 //org.springframework.boot.autoconfigure.EnableAutoConfiguration 的信息。 //点进去通过"SpringFactoriesLoader"进行加载 List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes); // removeDuplicates方法的作用是 移除同名的 configurations = removeDuplicates(configurations); // 获取我们配置的 exclude 信息 // 比如:@SpringBootApplication(exclude = {RabbitAutoConfiguration.class}) ,显示的指定不要加载那个配置类 Set<String> exclusions = getExclusions(annotationMetadata, attributes); checkExcludedClasses(configurations, exclusions); configurations.removeAll(exclusions); // filter的作用是 过滤掉咱们不需要使用的配置类。 configurations = getConfigurationClassFilter().filter(configurations); fireAutoConfigurationImportEvents(configurations, exclusions); return new AutoConfigurationEntry(configurations, exclusions); }
5.前面几个都好理解,现在我们主要看看filter(),是怎么移除不需要的类
我们可以看到有具体的匹配方法 match。里面有个关键的属性是 autoConfigurationMetadata , 的本质是 加载的 META-INF/spring-autoconfigure-metadata.properties 的文件中的内容。
其实原理很简单,如果没有对应的实现类,就不进行加载。
到这里自动装配的原理就完事了~三. 何时进行自动装配
四. run 方法
@SpringBootApplication public class SpringBootVipDemoApplication { public static void main(String[] args) { // 基于配置文件的方式 ApplicationContext ac1 = new ClassPathXmlApplicationContext(""); // 基于Java配置类的方式 ApplicationContext ac2 = new AnnotationConfigApplicationContext(SpringBootVipDemoApplication.class); // run 方法的返回对象是 ConfigurableApplicationContext 对象, //ConfigurableApplicationContext就是ApplicationContext的一个子接口 ConfigurableApplicationContext ac3 = SpringApplication.run(SpringBootVipDemoApplication.class, args); } }
所以我们发现SpringBoot项目的启动,本质上就是Spring的初始化操作。
想亲身感受自动装配,可以参考手写一个SpringBoot starter
以上是脚本宝典为你收集整理的SpringBoot自动装配全部内容,希望文章能够帮你解决SpringBoot自动装配所遇到的问题。
本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。