springboot自动装配流程源码讲解(以REDIS为例)

2 篇文章 0 订阅

springboot自动装配流程图.png

  1. springboot找到了@EnableAutoConfiguration注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration //开启自动配置注解
@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
		@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
  1. 使用@Import注解导入了AutoConfigurationImportSelector类
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)  //导入AutoConfigurationImportSelector.class
public @interface EnableAutoConfiguration {
  1. AutoConfigurationImportSelector调用selectImports()方法导入jar包下的meta-info/spring.factories找到所有EnableAutoConfiguration为key的集合,根据Maven依赖导入的jar来筛选配置类
@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());
	}

protected AutoConfigurationEntry getAutoConfigurationEntry(AutoConfigurationMetadata autoConfigurationMetadata,
			AnnotationMetadata annotationMetadata) {
		if (!isEnabled(annotationMetadata)) {
			return EMPTY_ENTRY;
		}
		AnnotationAttributes attributes = getAttributes(annotationMetadata);
		List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
///去除重复的配置类,若我们自己写的starter 可能存主重复的
		configurations = removeDuplicates(configurations);
		Set<String> exclusions = getExclusions(annotationMetadata, attributes);
		checkExcludedClasses(configurations, exclusions);
		configurations.removeAll(exclusions);
//根据maven 导入的启动器过滤出 需要导入的配置类
		configurations = filter(configurations, autoConfigurationMetadata);
		fireAutoConfigurationImportEvents(configurations, exclusions);
		return new AutoConfigurationEntry(configurations, exclusions);
	}

//导入 META-INF/spring.factories文件,查找对应的EnableAutoConfiguration配置

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;
	}

//进入SpringFactoriesLoader.loadFactoryNames 方法可以看到loadSpringFactories方法

private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
		MultiValueMap<String, String> result = cache.get(classLoader);
		if (result != null) {
			return result;
		}

		try {
			Enumeration<URL> urls = (classLoader != null ?
					classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
					ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
			result = new LinkedMultiValueMap<>();
			while (urls.hasMoreElements()) {
				URL url = urls.nextElement();
				UrlResource resource = new UrlResource(url);
				Properties properties = PropertiesLoaderUtils.loadProperties(resource);
				for (Map.Entry<?, ?> entry : properties.entrySet()) {
					String factoryTypeName = ((String) entry.getKey()).trim();
					for (String factoryImplementationName : StringUtils.commaDelimitedListToStringArray((String) entry.getValue())) {
						result.add(factoryTypeName, factoryImplementationName.trim());
					}
				}
			}
			cache.put(classLoader, result);
			return result;
		}
		catch (IOException ex) {
			throw new IllegalArgumentException("Unable to load factories from location [" +
					FACTORIES_RESOURCE_LOCATION + "]", ex);
		}
	}

image.png

  1. 因为导入了spring-boot-starter-data-redis的Maven依赖,因此RedisAutoConfiguration配置类会起作用(利用@ ConditionalOnClass注解)
@Configuration(proxyBeanMethods = false)
//当引入到RedisOperation类,配置才会生效
@ConditionalOnClass(RedisOperations.class)
@EnableConfigurationProperties(RedisProperties.class)
@Import({ LettuceConnectionConfiguration.class, JedisConnectionConfiguration.class })
public class RedisAutoConfiguration {

	@Bean
	@ConditionalOnMissingBean(name = "redisTemplate")
	public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory)
			throws UnknownHostException {
		RedisTemplate<Object, Object> template = new RedisTemplate<>();
		template.setConnectionFactory(redisConnectionFactory);
		return template;
	}

	@Bean
	@ConditionalOnMissingBean
	public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory redisConnectionFactory)
			throws UnknownHostException {
		StringRedisTemplate template = new StringRedisTemplate();
		template.setConnectionFactory(redisConnectionFactory);
		return template;
	}

}
  1. @EnableConfigurationProperties(RedisProperties.class)开启读取RedisProperties映射的配置,进入该注解查看源码可以看到引入了EnableConfigurationPropertiesRegistrar.class类
    image.png
    EnableConfigurationPropertiesRegistrar.class类实现了ImportBeanDefinitionRegistrar接口的类将会调用它的registerBeanDefinitions方法,向Spring容器中注册Bean定义
class EnableConfigurationPropertiesRegistrar implements ImportBeanDefinitionRegistrar {

	@Override
	public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
		registerInfrastructureBeans(registry);
		ConfigurationPropertiesBeanRegistrar beanRegistrar = new ConfigurationPropertiesBeanRegistrar(registry);
		getTypes(metadata).forEach(beanRegistrar::register);
	}

	private Set<Class<?>> getTypes(AnnotationMetadata metadata) {
		return metadata.getAnnotations().stream(EnableConfigurationProperties.class)
				.flatMap((annotation) -> Arrays.stream(annotation.getClassArray(MergedAnnotation.VALUE)))
				.filter((type) -> void.class != type).collect(Collectors.toSet());
	}

	@SuppressWarnings("deprecation")
	static void registerInfrastructureBeans(BeanDefinitionRegistry registry) {
		ConfigurationPropertiesBindingPostProcessor.register(registry);
		ConfigurationPropertiesBeanDefinitionValidator.register(registry);
		ConfigurationBeanFactoryMetadata.register(registry);
	}

}

@ConfigurationProperties可以将外部配置文件(applicaition.properties中spring.redis下的配置)加载进来,填充对象的对应字段的数据,然后供其他Bean使用。

/**
 * Configuration properties for Redis.
 *
 * @author Dave Syer
 * @author Christoph Strobl
 * @author Eddú Meléndez
 * @author Marco Aust
 * @author Mark Paluch
 * @author Stephane Nicoll
 * @since 1.0.0
 */
 //指明spring.redis作为前缀
@ConfigurationProperties(prefix = "spring.redis")
public class RedisProperties {

	/**
	 * Database index used by the connection factory.
	 */
	private int database = 0;

	/**
	 * Connection URL. Overrides host, port, and password. User is ignored. Example:
	 * redis://user:password@example.com:6379
	 */
	private String url;

	/**
	 * Redis server host.
	 */
	private String host = "localhost";

	/**
	 * Login password of the redis server.
	 */
	private String password;
  1. RedisAutoConfiguration里配置了redisTemplate和stringRedisTemplate组件
    image.png
  2. 使用@Import 注解导入了JedisConnectionConfiguration.class
    image.png
  3. JedisConnectionConfiguration 类把JedisConnectionFactory注册到了bean,并且配置了jedisPoolConfig
private JedisClientConfiguration getJedisClientConfiguration(
			ObjectProvider<JedisClientConfigurationBuilderCustomizer> builderCustomizers) {
		JedisClientConfigurationBuilder builder = applyProperties(JedisClientConfiguration.builder());
		RedisProperties.Pool pool = getProperties().getJedis().getPool();
		if (pool != null) {
			applyPooling(pool, builder);
		}
		if (StringUtils.hasText(getProperties().getUrl())) {
			customizeConfigurationFromUrl(builder);
		}
		builderCustomizers.orderedStream().forEach((customizer) -> customizer.customize(builder));
		return builder.build();
	}
private JedisPoolConfig jedisPoolConfig(RedisProperties.Pool pool) {
		JedisPoolConfig config = new JedisPoolConfig();
		config.setMaxTotal(pool.getMaxActive());
		config.setMaxIdle(pool.getMaxIdle());
		config.setMinIdle(pool.getMinIdle());
		if (pool.getTimeBetweenEvictionRuns() != null) {
			config.setTimeBetweenEvictionRunsMillis(pool.getTimeBetweenEvictionRuns().toMillis());
		}
		if (pool.getMaxWait() != null) {
			config.setMaxWaitMillis(pool.getMaxWait().toMillis());
		}
		return config;
	}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答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个

红包金额最低5元

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

抵扣说明:

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

余额充值