Java研学-SpringBoot(三)

五 Spring Boot 自动配置原理

1 概念

  springboot的自动装配就是从spring.factories文件中获取到对应的需要进行自动装配的类,并生成相应的Bean对象,然后将它们交给spring容器来帮我们进行管理。核心注解:@SpringBootApplication
  调用main函数之前会扫描该类上是否有注解,有注解就执行对应的某些功能SpringApplication.run(DemoApplication.class, args);表示把这个类加载进来作为主启动类,同时让springboot启动Tomcat,再将外界传递的参数通过args获取到(例如启动springboot项目时的命令 java -jar xxx.jar --server.port=80 就是通过args传递到项目中的)

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

2 @SpringBootApplication 核心注解(复合注解)

// 元注解
// 贴在哪里
@Target(ElementType.TYPE)
// 什么时期生效
@Retention(RetentionPolicy.RUNTIME)
// 文档标记
@Documented
// 继承
@Inherited

// 同为复合注解 含元注解中的123 与 @Configuration(装配类注解)
@SpringBootConfiguration

// 核心注解中的核心注解 自动装配
@EnableAutoConfiguration

// 扫描时 排除某些文件 从当前位置开始往下扫描
@ComponentScan(excludeFilters = {
	 @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
	 @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })

3 @EnableAutoConfiguration 核心注解(复合注解)

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
// 自动导入配置包
@AutoConfigurationPackage
// 导入AutoConfigurationImportSelector类 所有自动装配的活都是这个类做(自动配置类)
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
	String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";

	Class<?>[] exclude() default {};

	String[] excludeName() default {};

}

4 AutoConfigurationImportSelector.class

  于该类中的此方法中进行自动装配,基于注解元数据获取并筛选自动配置类,同时考虑用户可能希望排除的配置类。最后,它返回一个包含筛选后的配置和排除的配置的AutoConfigurationEntry对象

// 这是一个受保护的方法,返回AutoConfigurationEntry对象。它接收一个AnnotationMetadata对象作为参数,这个对象通常用于描述注解的元数据。
protected AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
		// 首先检查给定的注解元数据是否启用了自动配置。如果没有启用,则返回一个空的AutoConfigurationEntry
		if (!isEnabled(annotationMetadata)) {
			return EMPTY_ENTRY;
		}
		// 从注解元数据中获取注解属性,并将其存储在AnnotationAttributes对象中。
		AnnotationAttributes attributes = getAttributes(annotationMetadata);
		// 根据注解元数据和其属性,获取候选的自动配置类列表(127个结尾为AutoConfiguration的候选可装配类,版本不同数量不同)
		List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
		// 从候选配置列表中移除重复的配置。
		configurations = removeDuplicates(configurations);
		// 从注解元数据和其属性中,获取用户希望排除的自动配置类集合
		Set<String> exclusions = getExclusions(annotationMetadata, attributes);
		// 检查是否存在任何在候选配置列表中但又被排除的配置类。是一个检查或警告机制,确保用户不会错误地排除必要的配置
		checkExcludedClasses(configurations, exclusions);
		// 从候选配置列表中移除所有被排除的配置类。
		configurations.removeAll(exclusions);
		// 使用配置类过滤器进一步过滤候选配置列表。这通常用于根据特定条件(如条件注解)进一步筛选配置。
		configurations = getConfigurationClassFilter().filter(configurations);
		// 触发与自动配置导入相关的事件。这通常用于通知监听器或框架的其他部分,自动配置已经完成或即将完成
		fireAutoConfigurationImportEvents(configurations, exclusions);
		// 最后,返回一个新的AutoConfigurationEntry对象,该对象包含筛选后的配置列表和排除的配置列表
		return new AutoConfigurationEntry(configurations, exclusions);
	}

5 getCandidateConfigurations 方法

  通过SpringFactoriesLoader的loadFactoryNames方法获取这127个候选可装配类,这是一个受保护的方法,它接收两个参数:AnnotationMetadata(注解元数据)和AnnotationAttributes(注解属性)。该方法返回一个List,其中包含自动配置类的名称

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
// 这行代码调用了SpringFactoriesLoader的loadFactoryNames方法,该方法用于从META-INF/spring.factories文件中加载指定类型的工厂类名
		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;
	}

6 SpringFactoriesLoader.loadFactoryNames 方法

  用于从META-INF/spring.factories文件中加载特定类型的工厂名称列表,并返回这个列表。如果文件中没有指定类型的条目,则返回一个空列表

public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
// 这行代码获取factoryType的完全限定名(包括包名)
        String factoryTypeName = factoryType.getName();
// 拿到127个候选可装配类
        return (List)loadSpringFactories(classLoader).getOrDefault(factoryTypeName, Collections.emptyList());
    }

7 loadSpringFactories 方法

  这个方法的目的是从META-INF/spring.factories文件中读取并解析工厂条目,并将结果存储在缓存中以便后续快速访问。这是Spring Boot自动配置机制的一部分,它允许第三方库通过spring.factories文件来声明它们提供的自动配置类或其他工厂类。

// 定义了一个私有静态方法loadSpringFactories,它接收一个可能为null的类加载器classLoader作为参数,并返回一个映射,其中键是工厂类型的名称,值是相应类型的工厂实现类名的列表。
private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
// 尝试从缓存cache中获取已加载的spring.factories映射。这里假设cache是一个ConcurrentMap,其键是类加载器,值是MultiValueMap。
        MultiValueMap<String, String> result = (MultiValueMap)cache.get(classLoader);
        // 如果缓存中已经存在结果,则直接返回这个缓存的结果,避免重复加载。
        if (result != null) {
            return result;
        } else {
            try {
            // 尝试使用提供的类加载器(如果存在)或系统类加载器来获取所有META-INF/spring.factories资源的URL。
                Enumeration<URL> urls = classLoader != null ? classLoader.getResources("META-INF/spring.factories") : ClassLoader.getSystemResources("META-INF/spring.factories");
                // 初始化一个新的LinkedMultiValueMap来存储加载的工厂条目。这个映射允许同一个键关联多个值。
                LinkedMultiValueMap result = new LinkedMultiValueMap();

				// 遍历所有找到的spring.factories文件URL,为每个URL创建一个UrlResource,
				// 并使用PropertiesLoaderUtils加载文件内容为一个Properties对象。
                while(urls.hasMoreElements()) {
               		// 检索出的本地仓路径拼接上META-INF/spring.factories
                    URL url = (URL)urls.nextElement();
                    // urls实质上就是External Libraries中导入的所有jar包(jar包都在本地仓中),于程序中拿到所有的jar包,看每个jar包中是否有META-INF/spring.factories文件(不是每个jar包都有这个文件,带有start的依赖对应的jar包才会有),若存在将其中的内容(自动装配类的全名)读取出来,将这些自动装配类的全名(格式为xxxAutoConfiguration)存到一个List集合中作为候选配置(不一定会生效)
                    UrlResource resource = new UrlResource(url);
                    Properties properties = PropertiesLoaderUtils.loadProperties(resource);
                    // 遍历Properties对象的每个条目。每个条目的键是工厂类型名称,值是逗号分隔的工厂实现类名列表
                    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());
                        // 对于每个工厂类型名称,遍历其关联的工厂实现类名数组,并将实现类名添加到result映射中,与工厂类型名称相关联。
                        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) {
            // 如果在加载过程中出现IOException,则捕获该异常并抛出一个带有更具体消息的IllegalArgumentException
                throw new IllegalArgumentException("Unable to load factories from location [META-INF/spring.factories]", var13);
            }
        }
    }

8 WebMvcAutoConfiguration – 例子

// 注解
@Configuration(proxyBeanMethods = false)
// 必须满足某些条件 才能进行自动装配(Conditional 有条件的) 不满足则不装配
// 检查当前应用是否是一个基于 Servlet 的 Web 应用,是则进行装配
@ConditionalOnWebApplication(type = Type.SERVLET)
// 必须有以下3个类(需要spring-boot-starter-web依赖)
@ConditionalOnClass({ Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class })
// 当没有WebMvcConfigurationSupport这个Bean的时候进行自动装配,有则不进行装配
// WebMvcConfigurationSupport该类中会做所有的默认配置 若更改了默认配置 则不对更改的配置做默认配置了
// 即该类中都是可以自行传递的配置,若自己没配则全部使用默认配置(不满足于默认装配可自行配置(重写方法))
@ConditionalOnMissingBean(WebMvcConfigurationSupport.class)
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE + 10)
// 想使用SpringMvc需在配置DispatcherServlet之后
@AutoConfigureAfter({ DispatcherServletAutoConfiguration.class, TaskExecutionAutoConfiguration.class,
		ValidationAutoConfiguration.class })

9 小结

  Spring Boot的自动装配原理是基于其内部的一套约定和机制,使得开发者能够更便捷地构建Spring应用程序,而无需进行大量的手动配置。

  主方法中通过SpringApplication.run(DemoApplication.class, args)方法将这个类加载进来作为主启动类,同时让springboot启动Tomcat,再将外界传递的参数通过args获取到,而调用main函数之前就会进行扫描,该类上是否有注解,有注解就会执行对应的某些功能,在主方法之上的就是@SpringBootApplication注解

  该注解为复合注解包含了4个基本元注解,以及3个非元注解@SpringBootConfiguration(同为复合注解 含元注解中的123 与 @Configuration),@EnableAutoConfiguration(核心注解中的核心注解 自动装配),@ComponentScan(扫描时 排除某些文件 从当前位置开始往下扫描)

  其中@EnableAutoConfiguration注解中导入了一个 AutoConfigurationImportSelector 配置类,该类中有个 getCandidateConfigurations 方法,方法的作用是委托 SpringFactoriesLoader 的方法loadSpringFactories去读取 jar 包中的 META-INF/spring.factories 文件,并加载里面配置的自动配置对象,包括:AOP,Jackson,DataSourceDataSourceTransactionManager,DispatcherServlet 等等。

  选取其中一个对象,例如 Spring MVC,查看其自动配置类,类上含有4个注解@Configuration:声明这个类是一个配置类
  @ConditionalOnWebApplication(type = Type.SERVLET),ConditionalOn,翻译就是在某个条件下,此处就是满足项目的类是是 Type.SERVLET 类型
  @ConditionalOnClass({Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class })
这里的条件是 OnClass,也就是满足以下类存在:Servlet、DispatcherServlet、WebMvcConfigurer。这里就是判断是否引入了 Spring MVC相关依赖,引入依赖后该条件成立,当前类的配置才会生效!

  @ConditionalOnMissingBean(WebMvcConfigurationSupport.class)这个条件与上面不同,OnMissingBean,是说环境中没有指定的 bean 这个才生效。其实这就是自定义配置的入口,也就是说,如果我们自己配置了一个 WebMVCConfigurationSupport 的 bean,代表容器里已经存在该 bean 了,那么这个默认配置就会失效!@AutoConfigureAfter 注解,意为指定的类加载完了后,再加载本类。

  项目中添加了很多jar,在这些jar包中,我们依次去寻找spring.factories文件,该文件中存在了自动装配类的全名,这些类的名字有一个特点都是以XxxAutoConfiguration,将这些名字存到一个List集合中作为候选配置(所谓的候选就是不一定会生效)

  至于这些自动装配类是否生效取决于,XxxAutoConfiguration类上面对注解,@ConditionalOnXxx我们称之为条件注解,当有某些依赖的时候,自动装配生效,当相遇中有需求需要根据自己配置的时候,会让其自动装配不生效(大改)

  自动从xxx.properties获取配置,获取的具体内容可以,通过application.properties或者application.yml,配置文件中修改(小改)

  若自动装配生效,本质上其实就是在帮我们创建一些,Bean 存到容器中。

  PS:对于依赖启动器,找到其对应的Maven:xxx-autoconfigure文件中的 META-INF/spring.factories 文件,其中就是其对应的自动装配候选类,选中定位后下面一般会对应有一个properties文件,其中有对应的前缀,获得前缀后就可以到application.properties文件中通过前缀进行配置,完成装配

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/499592.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

20232831 2023-2024-2 《网络攻防实践》第4次作业

目录 20232831 2023-2024-2 《网络攻防实践》第4次作业1.实验内容2.实验过程&#xff08;1&#xff09;ARP缓存欺骗攻击&#xff08;2&#xff09;ICMP重定向攻击&#xff08;3&#xff09;SYN Flood攻击&#xff08;4&#xff09;TCP RST攻击&#xff08;5&#xff09;TCP会话…

为什么你的企业需要渗透测试

渗透测试是什么&#xff1f; 渗透测试是在安全、符合规定并且受控的条件下针对公司精心策划的经过批准的网络攻击。渗透测试人员努力发现和利用组织环境的设定范围内的漏洞&#xff0c;在黑客等犯罪分子利用它们之前提前分析弱点。 渗透测试通常是安全审计的一部分&#xff0…

大规模云存储展望|2024逐步复苏,2025全面恢复

SSD以其高速度和低延迟等优点&#xff0c;尤其在容量增长和每GB成本降低方面&#xff0c;SSD的增长速度预计将超过近线硬盘&#xff08;Nearline HDD&#xff09;。尽管HDD在大容量存储方面仍有一定优势&#xff0c;但由于SSD在访问速度、能耗及体积等方面的突出表现&#xff0…

fastapi学习记录

今天看了点fastap&#xff0c;简单记录下&#xff0c;fastapi是一个python下的后端框架。 参考学习网站菜鸟教程 安装 pip install fastapi pip install "uvicorn[standard]"安装好了以后就可以直接使用&#xff0c;最主要的使用方式就是写接口嘛&#xff0c;get&a…

FMEA与智能机器人:提升机器人可靠性与安全性的关键

随着科技的飞速发展&#xff0c;智能机器人已经深入到我们生活的方方面面&#xff0c;从工业生产到家庭服务&#xff0c;从深海探险到太空探索&#xff0c;处处都有它们的身影。然而&#xff0c;随着应用的日益广泛&#xff0c;机器人系统的复杂性和不确定性也在增加&#xff0…

速成软件书是神器还是焦虑?

一、背景 "速成软件书"通常是指那些宣称能帮助读者在短时间内掌握某种软件操作或编程技能的书籍。这类书籍往往以其高效、快捷的学习路径吸引读者&#xff0c;尤其适合有一定基础或者急需短期内提升特定技能的人群。 然而&#xff0c;“神器”之称则带有主观性和一…

双端队列deque和vector以及list的优缺点比较

参考:https://blog.csdn.net/TWRenHao/article/details/123483085 一、vector vector具体用法详情点这里 优点&#xff1a; 支持随机访问 CPU高速环缓存命中率很高 缺点&#xff1a; 空间不够&#xff0c;便需要增容。而增容代价很大&#xff0c;还存在一定的空间浪费。 头部…

数据可视化为什么能在智慧港口中发挥作用?

随着全球贸易活动日益频繁&#xff0c;港口作为国际贸易的重要节点&#xff0c;其运营效率与智能化程度直接影响着整个物流链的效能。在此背景下&#xff0c;智慧港口的概念应运而生&#xff0c;它借助先进的信息技术手段对传统港口进行改造升级&#xff0c;其中&#xff0c;数…

基于Arduino IDE 野火ESP8266模块 文件系统LittleFS 的开发

一、文件系统LittleFS的介绍 LittleFS是一个为微控制器设计的轻量级、可靠且高性能的文件系统。它专为嵌入式设备打造&#xff0c;拥有占用空间小、对硬件要求低的特点&#xff0c;同时保证在断电情况下数据的完整性和稳定性。 1.设计与特点 LittleFS的设计旨在提供嵌入式系统所…

第三十二天-Django模板-DTL模板引擎

目录 1.介绍 2. 使用 1.配置jinja2 2.DTL模板变量使用 3.与jinja2区别 4.模板标签使用 1.循环 2.条件控制 3.注释 4.url解析 5.显示时间 5.模板的基础与包含 6.过滤器 内置过滤器 自定义过滤器 1.介绍 2. 使用 1.配置jinja2 2.DTL模板变量使用 与jinja2语法相似…

PHP图床程序优化版:图片外链服务、图床API服务、图片CDN加速与破解防盗链

图片免费上传 支持本地储存、FTP储存、第三方云储存&#xff08;阿里云 OSS、腾讯云 COS、七牛云等&#xff09;。 图片外链加速 一键转换第三方网站的图片外链地址为图床可分享的图片地址&#xff08;支持CDN&#xff09;。 图片解析服务 直接将第三方外链图片地址显示为…

Linux网络配置(超详细)

Linux网络配置大全 Linux网络配置一.网络地址配置网络地址查看–ifconfig使用网络配置命令设置网络接口参数-ifconfig禁用(临时)或者重新激活网卡设置虚拟网络接口 修改网络配置文件网络接口配置文件 IP命令详解OPTIONS选项OBJECT对象 ip link 二、获取和修改主机名hostname查看…

数对 离散化BIT

先把公式变个形&#xff0c;然后直接BIT 枚举右端点查询左端点累加答案 离散化好题&#xff0c;注意BIT写的时候右端点的范围是离散化区间的大小 #include<bits/stdc.h> using namespace std; #define int long long using ll long long; using pii pair<int,int&…

【IC前端虚拟项目】write_path子模块DS与RTL编码

【IC前端虚拟项目】数据搬运指令处理模块前端实现虚拟项目说明-CSDN博客 read_path的代码完成之后,就可以开始整个项目里复杂度最高、bug最多、时序收敛最为困难的模块——write_path的开发了!我自己写过两次这个虚拟项目,每次都是在这里耗时最久,所以大家也可以挑战一下自…

MAC上好用的文件查找软件

和windows上的everything很像&#xff0c;不过就是要钱&#xff0c;我简单测试了一下还可以蛮好用的&#xff0c;

【QT入门】 QTabWidget各种常见用法详解

往期回顾&#xff1a; 【QT入门】 Qt代码创建布局之分裂器布局详解-CSDN博客 【QT入门】 Qt代码创建布局之setLayout使用-CSDN博客 【QT入门】 Qt代码创建布局之多重布局变换与布局删除技巧-CSDN博客 【QT入门】 QTabWidget各种常见用法详解 一般来说&#xff0c;学一个新的控…

关系型数据库mysql(7)sql高级语句①

目录 一.MySQL常用查询 1.按关键字&#xff08;字段&#xff09;进行升降排序 按分数排序 &#xff08;默认为升序&#xff09; 按分数升序显示 按分数降序显示 根据条件进行排序&#xff08;加上where&#xff09; 根据多个字段进行排序 ​编辑 2.用或&#xff08;or&…

以新质生产力引领智能锁行业腾飞,凯迪仕打造全球最大智能安防产业园

凯迪仕&#xff0c;作为智能锁行业的领军企业&#xff0c;今日在温州举行了凯迪仕全球超级工厂落成庆典。积极拥抱新质生产力&#xff0c;大力发展智能制造&#xff0c;凯迪仕在全球制造业科技创新的制高点上迈出了坚实的步伐。 浙江省温州市瓯海区委副书记、区长刘云峰&#x…

作者开发的爬取妹子图片Python项目,值得你收藏拥有

最好的学习方法在于实践&#xff0c;学习编程语言Python&#xff0c;也是同样的道理。本文讲解自己开发的一个项目&#xff0c;实现爬取妹子图片&#xff0c;所用的Python知识点以及模块&#xff0c;可以关注参考作者公众号的Python语言合集。 —、前情介绍 1.1 涉及模块 本…

泛微E-Office10 < 10.0_20240222 远程代码执行漏洞

一、软件背景 泛微e-office是一套企业级电子办公解决方案&#xff0c;提供文档管理、流程审批、协同办公等功能&#xff0c;帮助企业实现数字化办公、提高工作效率。 二、影响版本 e-office10[10.0_20180516, 10.0_20240222) 三、漏洞分析 在受影响版本中&#xff0c;由于…
最新文章