SpringBoot排除不需要的自动配置类DataSourceAutoConfiguration问题

 更新时间:2024年07月11日 14:38:02   作者:山河锦绣  
这篇文章主要介绍了SpringBoot排除不需要的自动配置类DataSourceAutoConfiguration问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

一、排除自动配置类的三种方式

以下三种方式可以用来排除任意的自动配置类

1.1使用@SpringBootApplication注解排除

使用exclude属性(value是Class对象数组)

@SpringBootApplication(exclude = DataSourceAutoConfiguration.class)
public class DemoApplication {

	public static void main(String[] args) {
		SpringApplication.run(DemoApplication.class, args);
	}

}

或者excludeName属性(value是类的全限定名字符串数组)

@SpringBootApplication(excludeName = {"org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration"})
public class DemoApplication {

	public static void main(String[] args) {
		SpringApplication.run(DemoApplication.class, args);
	}

}

1.2使用@EnableAutoConfiguration注解排除

1.在项目中创建一个配置类对象MyConfig.class,在配置类上使用@EnableAutoConfiguration注解。

使用exclude属性(value是Class对象数组)

@Configuration
@EnableAutoConfiguration(exclude = DataSourceAutoConfiguration.class)
public class MyConfig {
}

或者excludeName属性(value是类的全限定名字符串数组)

@Configuration
@EnableAutoConfiguration(excludeName = {"org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration"})
public class MyConfig {
}

思考1:为什么不能直接在启动类上使用@EnableAutoConfiguration注解,而要重新创建一个配置类?

因为@SpringBootApplication已经继承了@EnableAutoConfiguration,并且@EnableAutoConfiguration注解,是不可在类上重复使用的注解(换句话说就是使用@SpringBootApplication注解,就相当于使用了@EnableAutoConfiguration注解)

所以 我们会发现方法一与方法二本质上是一样的只是形式上的差别,底层都是依赖于@EnableAutoConfiguration注解实现的排除功能。

思考2:@SpringBootApplication注解与@EnableAutoConfiguration的关系?为什么在@SpringBootApplication注解中配置属性底层还是使用@EnableAutoConfiguration实现的呢?

@SpringBootApplication注解源码:

@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 {

	/**
	 * Exclude specific auto-configuration classes such that they will never be applied.
	 * @return the classes to exclude
	 */
	@AliasFor(annotation = EnableAutoConfiguration.class)
	Class<?>[] exclude() default {};

	/**
	 * Exclude specific auto-configuration class names such that they will never be
	 * applied.
	 * @return the class names to exclude
	 * @since 1.3.0
	 */
	@AliasFor(annotation = EnableAutoConfiguration.class)
	String[] excludeName() default {};

我们可以看出源码中使用了注解:

@AliasFor(annotation = EnableAutoConfiguration.class)

将@SpringBootApplication的exclude和excludeName属性与@EnableAutoConfiguration的属性做了一个桥接的作用。

1.3.在yml配置文件中添加排除配置

spring:
  autoconfigure:
    exclude: org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration

注:如果要排除多个类,使用逗号隔开

二、为什么可以这样排除,内在逻辑原理研究

springBoot将自动配置类添入到spring容器中的部分源码:

	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);
	}
  • 1.这段代码中的主要逻辑就是,将项目中所有需要的自动配置类的全限定名以字符串数组返回出去,后续spring容器会将所有的配置类创建成Bean
  • 2.所以我们三种配置排除自动配置类的行为,最终都会在这段代码逻辑中体现出来,在spring容器初始化他们之前,我们就已经提前把他们从初始化的名单中排除了
  • 3.我们配置的时候为什么都是使用的类的全限定名在这里也得到了解释,因为只有使用全限定名spring容器才能将其创建成对象。
  • 4.还可以得出结论就是几种方式可以同时使用,最终排除的是他们的并集

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Java后台实现浏览器一键导出下载zip压缩包

    Java后台实现浏览器一键导出下载zip压缩包

    这篇文章主要为大家详细介绍了Java后台实现浏览器一键导出下载zip压缩包,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • Java实现MinIO文件上传的加解密操作

    Java实现MinIO文件上传的加解密操作

    在云存储场景中,数据安全是核心需求之一,MinIO作为高性能对象存储服务,支持通过客户端加密(CSE)在数据上传前完成加密,下面我们来看看如何通过Java实现MinIO文件的加密上传与解密下载吧
    2025-05-05
  • struts2如何使用拦截器进行用户权限控制实例

    struts2如何使用拦截器进行用户权限控制实例

    本篇文章主要介绍了struts2如何使用拦截器进行用户权限控制实例,非常具有实用价值,需要的朋友可以参考下
    2017-05-05
  • 2018版java多线程面试题集合及答案

    2018版java多线程面试题集合及答案

    这篇文章主要为大家详细介绍了2018版java多线程面试题集合及答案,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-09-09
  • Mybatis中 XML配置详解

    Mybatis中 XML配置详解

    这篇文章主要介绍了Mybatis中 XML配置详解的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-01-01
  • java实现Object和Map之间的转换3种方式

    java实现Object和Map之间的转换3种方式

    本篇文章主要介绍了java实现Object和Map之间的转换3种方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • idea编译报错-代码没问题IDEA编译不通过的处理方案

    idea编译报错-代码没问题IDEA编译不通过的处理方案

    这篇文章主要介绍了idea编译报错-代码没问题IDEA编译不通过的问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • Bean的自动注入及循环依赖问题

    Bean的自动注入及循环依赖问题

    本文详细介绍了Bean的自动注入及循环依赖,文中通过代码介绍的非常详细,对大家的学习有一定的研究价值,感兴趣的小伙伴可以阅读参考
    2023-03-03
  • Java多线程之ReentrantReadWriteLock源码解析

    Java多线程之ReentrantReadWriteLock源码解析

    这篇文章主要介绍了Java多线程之ReentrantReadWriteLock源码解析,文中有非常详细的代码示例,对正在学习java基础的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-05-05
  • SpringBoot静态资源及原理解析

    SpringBoot静态资源及原理解析

    这篇文章主要介绍了SpringBoot静态资源及原理解析,当创建一个jar工程时,想引入css等静态资源时,需要遵守SpringBoot的静态资源映射关系,通过WebMvcAutoConfiguration查看静态配置资源的规则,需要的朋友可以参考下
    2023-12-12

最新评论