<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency>2、在 application.properties 文件中加入 Redis 相关的配置
spring.redis.host=127.0.0.1 spring.redis.port=63793、在代码中引用 Redis 缓存的操作类
@Autowired private RedisTemplate<String,String>redisTemplate;为什么 RedisTemplate 可以被直接注入,它是什么时候加入到 Ioc 容器中的,这都是自动装配的功劳,我们一起来看一下。
@AutoConfigurationPackage @Import(AutoConfigurationImportSelector.class) public @interface EnableAutoConfiguration { ... }1.1 @AutoConfigurationPackage
@Override public String[] selectImports(AnnotationMetadata annotationMetadata) { if (!isEnabled(annotationMetadata)) { return NO_IMPORTS; } //1、从配置文件spring-autoconfigure-metadata.properties中加载自动装配候选规则 AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader .loadMetadata(this.beanClassLoader); //2、获取@SpringBootApplication上配置的属性值 AnnotationAttributes attributes = getAttributes(annotationMetadata); //3、使用SpringFactoriesLoader 加载classpath路径下META-INF\spring.factories中 //通过key=org.springframework.boot.autoconfigure.EnableAutoConfiguration获取候选类 List<String> configurations = getCandidateConfigurations(annotationMetadata,attributes); //4、去除重复值 configurations = removeDuplicates(configurations); //5、获取exclude属性值,将exclude中的值排除掉 Set<String> exclusions = getExclusions(annotationMetadata, attributes); checkExcludedClasses(configurations, exclusions); configurations.removeAll(exclusions); //6、检查候选配置类上的注解@ConditionalOnClass,如果要求的类不存在,则这个候选类会被过滤不被加载 configurations = filter(configurations, autoConfigurationMetadata); fireAutoConfigurationImportEvents(configurations, exclusions); return StringUtils.toStringArray(configurations); }第一步和第三步逻辑中涉及到两个非常重要的文件 spring-autoconfigure-metadata.properties、spring.factories
@Import(PersonConfig.class) @Configuration public class PersonConfiguration { }引入 ImportSelector 实现类(AutoConfigurationImportSelector 就是实现了这个接口),这个接口需要实现的方法是 selectImports(),它返回的是一个字符串数组,代表的是类名集合,这些类将会被加载到Ioc容器中。
@Import(TestImportSelector.class) @Configuration public class ImportTestConfig { } public class TestImportSelector implements ImportSelector { @Override public String[] selectImports(AnnotationMetadata importingClassMetadata) { return new String[]{"com.example.service.TestService"}; } }引用 ImportBeanDefinitionRegistrar 实现类(AutoConfigurationPackages.Registrar 就是实现类这个接口),这个接口需要实现的方法是 registerBeanDefinitions(),它有两个入参,第一个参数 AnnotationMetadata代表当前类的注解信息;第二个参数 registry 代表的是 DefaultListableBeanFactory 实例,因为参数 DefaultListableBeanFactory 代表的是 Ioc 容器,如果想注入 Bean,可以直接对该类进行操作。
@Import(TestImportBeanDefinitorSelector.class) @Configuration public class ImportBeanDefinitionTestConfig { } public class TestImportBeanDefinitorSelector implements ImportBeanDefinitionRegistrar { @Override public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder .rootBeanDefinition(Person.class) .getBeanDefinition(); registry.registerBeanDefinition("person", beanDefinition); } }2.2 @Import实现原理
@Override public void refresh() throws BeansException, IllegalStateException { //....省略n行代码 //1.beanFactory后置处理逻辑,在这个方法里加载ConfigurationClassPostProcessor invokeBeanFactoryPostProcessors(beanFactory); //2.注册bean后置处理逻辑 registerBeanPostProcessors(beanFactory); //...省略n行代码 //3.实例化非懒加载的bean,并加入到Ioc容器中 finishBeanFactoryInitialization(beanFactory); //....省略n行代码 }
ConfigurationClassPostProcessor 实现了方法 postProcessBeanDefinitionRegistry(),在这个方法中跟踪代码到 ConfigurationClassParser.parse(),所有配置类的解析逻辑都在 parse() 方法中进行处理。
在 ConfigurationClassParser.parseparse() 继续往下跟踪会到 doProcessConfigurationClass() 方法,在该方法中会有一些常用配置注解的解析,例如 @Component、@ComponentScan、@Bean、@Configuration、@Import 等。
@Nullable protected final SourceClass doProcessConfigurationClass(ConfigurationClass configClass, SourceClass sourceClass) throws IOException { //...省略n行代码 //加载@Import注解,递归解析,获取导入的配置类 processImports(configClass, sourceClass, getImports(sourceClass), true); //...省略n行代码 } processImports() 中主要实现类了 @Import 接口的 3 种不同的加载方式 private void processImports(ConfigurationClass configClass, SourceClass currentSourceClass, Collection<SourceClass> importCandidates, boolean checkForCircularImports) { //...省略n行代码 if (candidate.isAssignable(ImportSelector.class)) { //1.实现了ImportSelector接口的类在@Import中引用逻辑 Class<?> candidateClass = candidate.loadClass(); ImportSelector selector = BeanUtils.instantiateClass( candidateClass,ImportSelector.class); ParserStrategyUtils.invokeAwareMethods( selector, this.environment, this.resourceLoader, this.registry); if (this.deferredImportSelectors != null && selector instanceof DeferredImportSelector) { this.deferredImportSelectors.add( new DeferredImportSelectorHolder(configClass, (DeferredImportSelector) selector)); } else { String[] importClassNames = selector.selectImports(currentSourceClass.getMetadata()); Collection<SourceClass> importSourceClasses = asSourceClasses(importClassNames); processImports(configClass, currentSourceClass, importSourceClasses, false); } } else if (candidate.isAssignable(ImportBeanDefinitionRegistrar.class)) { //2.实现了ImportBeanDefinitionRegistrar接口的类在@Import中引用逻辑 Class<?> candidateClass = candidate.loadClass(); ImportBeanDefinitionRegistrar registrar =BeanUtils.instantiateClass( candidateClass, ImportBeanDefinitionRegistrar.class); ParserStrategyUtils.invokeAwareMethods( registrar, this.environment, this.resourceLoader, this.registry); configClass.addImportBeanDefinitionRegistrar( registrar, currentSourceClass.getMetadata()); } else { //3.普通类直接在@Import中引用逻辑 this.importStack.registerImport( currentSourceClass.getMetadata(), candidate.getMetadata().getClassName()); processConfigurationClass(candidate.asConfigClass(configClass)); } // 堆代码 duidaima.com //...省略n行代码 }总结一下就是如下的方法链调用
refresh()=>invokeBeanFactoryPostProcessors()=>postProcessBeanDefinitionRegistry()=>parse()=> doProcessConfigurationClass()=>processImports()
从 SpringBoot 启动类的 run() 方法开始,跟踪代码到 SpringApplication.run() 方法,这里是 SpringBoot 启动的核心逻辑。在 SpringApplication.run() 方法中有一个 prepareContext() 方法,进入这个方法里面,会发现有一个 load() 方法,这里就是加载启动类的地方,它会将启动类注入到 Ioc 容器中。
private void prepareContext(ConfigurableApplicationContext context, ConfigurableEnvironment environment, SpringApplicationRunListeners listeners, ApplicationArguments applicationArguments, Banner printedBanner) { //...省略n行代码 //加载启动类,将启动类注入到Ioc容器中 load(context, sources.toArray(new Object[0])); //...省略n行代码 }在断点中可以看到,SpringBoot 启动类注入到了 annotatedReader 中(AnnotatedBeanDefinitionReader 基于注解的 beanDefinition 解析器),在这里将启动类加入到了 Ioc 容器。
run()=>prepareContext()()=>load()=>parse()=>register()