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) {// 首先检查给定的注解元数据是否启用了自动配置。如果没有启用,则返回一个空的AutoConfigurationEntryif (!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.factoriesURL 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,则捕获该异常并抛出一个带有更具体消息的IllegalArgumentExceptionthrow 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.mzph.cn/news/777253.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用 Yoda 和 ClickHouse 进行实时欺诈检测

背景 Instacart 是北美领先的在线杂货公司,拥有数百万活跃的客户和购物者。在其平台上打击欺诈和滥用行为不仅对于维护一个值得信赖和安全的环境至关重要,也对保持Instacart的财务健康至关重要。在这篇文章中,将介绍了一个欺诈平台——Yoda,解释了为什么我们选择ClickHous…

奖学金NOIP2007 普及组 T1

题目背景 NOIP2007 普及组 T1 题目描述 某小学最近得到了一笔赞助&#xff0c;打算拿出其中一部分为学习成绩优秀的前 55 名学生发奖学金。期末&#xff0c;每个学生都有 33 门课的成绩&#xff1a;语文、数学、英语。先按总分从高到低排序&#xff0c;如果两个同学总分相同…

Linux 反引号、单引号以及双引号的区别

1.单引号—— 单引号中所有的字符包括特殊字符&#xff08;$,,和\&#xff09;都将解释成字符本身而成为普通字符。它不会解析任何变量&#xff0c;元字符&#xff0c;通配符&#xff0c;转义符&#xff0c;只被当作字符串处理。 2.双引号——" 双引号&#xff0c;除了$,…

Android 14.0 SystemUI下拉状态栏增加响铃功能

1.概述 在14.0的系统产品rom定制化开发中,在对systemui的状态栏开发中,对SystemUI下拉状态栏的QuickQSPanel区域有快捷功能键开关,对于增加各种响铃快捷也是常用功能, 有需要增加响铃功能开关功能,接下来就来分析SystemUI下拉状态栏QuickQSPanel快捷功能键开关的相关源码…

API接口开发亚马逊国际获得AMAZON商品详情api采集商品详情页实时数据、实时销量、库存等参数接入演示

要获取亚马逊商品的实时数据、销量和库存等信息&#xff0c;您需要使用亚马逊提供的一些API接口。以下是一些可能的步骤&#xff1a; 注册key账号&#xff1a;首先&#xff0c;您需要注册一个账号。这将允许您访问亚马逊的各种API。 创建一个新的应用&#xff1a;在您的开发者…

AI智能分析网关智慧食安监管系统方案

3.15晚会刚过不久&#xff0c;淀粉肠的“屈辱”终于得以洗清&#xff0c;但某些品牌奶茶、梅菜扣肉、预制菜等等&#xff0c;生产过程仍是触目惊心。如何提升食品安全管理水平&#xff0c;保障食品从生产到消费环节的质量和安全&#xff1f;TSINGSEE青犀智利用智能分析网关V4Ea…

【图像合成】基于DCGAN典型网络的MNIST字符生成(pytorch)

关于 近年来&#xff0c;基于卷积网络&#xff08;CNN&#xff09;的监督学习已经 在计算机视觉应用中得到了广泛的采用。相比之下&#xff0c;无监督 使用 CNN 进行学习受到的关注较少。在这项工作中&#xff0c;我们希望能有所帮助 缩小了 CNN 在监督学习和无监督学习方面的成…

aws使用记录

数据传输&#xff08;S3) 安装命令行 安装awscli: https://docs.aws.amazon.com/zh_cn/cli/latest/userguide/getting-started-install.html#getting-started-install-instructions 直到 aws configure list 可以运行 身份验证&#xff1a; 运行&#xff1a; aws config…

字符计数.

题目描述 给定一个单词&#xff0c;请计算这个单词中有多少个元音字母&#xff0c;多少个辅音字母。 元音字母包括 a,e,i,o,u&#xff0c;共五个&#xff0c;其他均为辅音字母。 输入描述 输入格式 输入一行&#xff0c;包含一个单词&#xff0c;!单词中只包含小写英文字母。单…

【QGIS从shp文件中筛选目标区域导出为shp】

文章目录 1、写在前面2、QGIS将shp文件中目标区域输出为shp2.1、手动点选2.2、高级过滤 3、上述shp完成后&#xff0c;配合python的shp文件&#xff0c;即可凸显研究区域了 1、写在前面 利用shp文件制作研究区域mask&#xff0c;Matlab版本&#xff0c;请点击 Matlab利用shp文…

网络编程综合项目-多用户通信系统

文章目录 1.项目所用技术栈本项目使用了java基础&#xff0c;面向对象&#xff0c;集合&#xff0c;泛型&#xff0c;IO流&#xff0c;多线程&#xff0c;Tcp字节流编程的技术 2.通信系统整体分析主要思路&#xff08;自己理解&#xff09;1.如果不用多线程2.使用多线程3.对多线…

uniapp-Form示例(uviewPlus)

示例说明 Vue版本&#xff1a;vue3 组件&#xff1a;uviewPlus&#xff08;Form 表单 | uview-plus 3.0 - 全面兼容nvue的uni-app生态框架 - uni-app UI框架&#xff09; 说明&#xff1a;表单组建、表单验证、提交验证等&#xff1b; 截图&#xff1a; 示例代码 <templat…

3月28日,每日信息差

&#x1f384; 亚马逊未来15年或斥资1480亿美元在全球建设并运营数据中心 &#x1f30d; 中国移动正式商用5.5G 网络&#xff0c;Find X7系列成为首款5.5G手机 &#x1f30b; UC网盘官宣上传、下载不限速 &#x1f381; 优酷发布行业首款影视制作车 ✨ 阿里云联发科联手为手机芯…

O2OA(翱途)开发平台-快速入门开发一个门户实例

O2OA(翱途)开发平台[下称O2OA开发平台或者O2OA]拥有门户页面定制与集成的能力&#xff0c;平台通过门户定制&#xff0c;可以根据企业的文化&#xff0c;业务需要设计符合企业需要的统一信息门户&#xff0c;系统首页等UI界面。本篇主要介绍通过门户管理系统如何快速的进行一个…

第一次开发微信小程序的总结与心得

我们的小程序开发团队有三个人——两个前端和一个后端&#xff0c;一个没有产品经理和UI设计师的队伍&#xff0c;一个小程序开发零经验的队伍&#xff0c;却需要完成需求分析、页面设计、代码编写、功能测试、小程序上线的整个过程&#xff0c;所以在开发过程中&#xff0c;我…

团队建设与管理案例分析题

习题一 最近A公司接了一个信息系统运维项目&#xff0c;而且非常重视&#xff0c;任命了有丰富售后服务张某为系统规划与管理师&#xff0c;全权授权张某负责该项目&#xff0c;并要求他负责企业运维服务能力建设和提升。张某也学习了大量项目管理知识和运维管理知识&#xff…

刷题有一个疑问

public class Main {public static void main(String[] args) { // boolean a zhengchu(2,10); // System.out.print(a); // boolean a btrue(126); // System.out.print(a);int decimalNumber 10; // 十进制数String binaryString Integer.toS…

学点儿Java_Day12_IO流

1 IO介绍以及分类 IO: Input Output 流是一组有顺序的&#xff0c;有起点和终点的字节集合&#xff0c;是对数据传输的总称或抽象。即数据在两设备间的传输称为流&#xff0c;流的本质是数据传输&#xff0c;根据数据传输特性将流抽象为各种类&#xff0c;方便更直观的进行数据…

C++取经之路(其二)——含数重载,引用。

含数重载: 函数重载是指&#xff1a;在c中&#xff0c;在同一作用域&#xff0c;函数名相同&#xff0c;形参列表不相同(参数个数&#xff0c;或类型&#xff0c;或顺序)不同&#xff0c;C语言不支持。 举几个例子&#xff1a; 1.参数类型不同 int Add(int left, int right)…

【任职资格】某大型制造型企业任职资格体系项目纪实

该企业以业绩、责任、能力为导向&#xff0c;确定了分层分类的整体薪酬模式&#xff0c;但是每一名员工到底应该拿多少工资&#xff0c;同一个岗位的人员是否应该拿同样的工资是管理人员比较头疼的事情。华恒智信顾问认为&#xff0c;通过任职资格评价能实现真正的人岗匹配&…