SpringBoot自动装配定义先后顺序失效原因极其解析
SpringBoot自动装配定义先后顺序失效原因极其解析
- 1、场景分析
- 1.1、问题总结
- 2、使用`@AutoConfigureBefore`、`@AutoConfigureAfter`和`@AutoConfigureOrder`注解指定加载顺序
- 2.2、@AutoConfigureXX注解失效原因总结
- 3、使用静态内部装配类提升加载顺序
- 4、bean加载顺序规则
1、场景分析
遇到的场景:
最近写定义依赖时,需要结合SpringMvc的WebMvcConfigurationSupport扩展异常解析器方法extendHandlerExceptionResolvers
@Beanpublic HandlerExceptionResolver handlerExceptionResolver(@Qualifier("mvcContentNegotiationManager") ContentNegotiationManager contentNegotiationManager) {List<HandlerExceptionResolver> exceptionResolvers = new ArrayList<>();configureHandlerExceptionResolvers(exceptionResolvers);if (exceptionResolvers.isEmpty()) {addDefaultHandlerExceptionResolvers(exceptionResolvers, contentNegotiationManager);}extendHandlerExceptionResolvers(exceptionResolvers);HandlerExceptionResolverComposite composite = new HandlerExceptionResolverComposite();composite.setOrder(0);composite.setExceptionResolvers(exceptionResolvers);return composite;}
其中extendHandlerExceptionResolvers方法提供了可继承的接口,让开发者能够添加自定义的异常解析器放到HandlerExceptionResolverComposite中供dispatchServlet解析异常
1.1、问题总结
自定义的异常解析器配置类继承WebMvcConfigurationSupport类重写extendHandlerExceptionResolvers方法后,按照开放接口的原理,我们的异常解析器会添加到HandlerExceptionResolverComposite列表中,但是实际上并非如此。
- 开发者自定义的装配类和
WebMvcConfigurationSupport引用的jar包中装配类顺序 - 解决方案一:使用
@AutoConfigureBefore、@AutoConfigureAfter和@AutoConfigureOrder注解指定加载顺序 - 解决方案二:使用静态内部装配类提前加载
2、使用@AutoConfigureBefore、@AutoConfigureAfter和@AutoConfigureOrder注解指定加载顺序
SpringBoot下可以通过@Configuration自动扫描配置类和spring.factories来加载配置类,但这两种方式都无法控制加载顺序。
此时,可通过在配置类上增加@AutoConfigureAfter 、 @AutoConfigureBefore和@AutoConfigureOrder来控制配置文件加载的相对顺序。
SpringBoot的自动配置是通过spring.factories来指定的,它的优先级最低,加载时间最晚,spring.factories中的配置类顺序不代表实际加载顺序。可结合 @AutoConfigureAfter 和 @AutoConfigureBefore注解控制配置类的相对加载顺序。
通过@Configuration和@ComponentScan扫描加载的配置类,一般是我们自定义的配置类,这部分配置类优先级最高,加载时间最早,在加载spring.factories配置类前加载,但加载顺序不定。
这里就存在另一个易错点。简单的理解,当配置类在Spring扫描路径里面(scanBasePackages)会优先解析,后面在通过ImportSelector(spring.factories加载就是通过实现这个接口加载的配置类)加载进来的配置类就不会处理了,相当于一个类有两种加载方式,谁先加载谁就厉害。。。这里加载都会调用到processConfigurationClass()方法,这个下面会说!!!!
空口无凭上菜:ConfigurationClassParser–>doProcessConfigurationClass()方法加载顺序是@ComponentScan(扫描文件路径,路径里面元注解为@Component(@Cofiguration元注解也是@Component)都会被扫描到)—>加载@Import注解(配合ImportSelector接口)—>加载 @ImportResource—>加载@Bean—>…
大致顺序理清楚了。
也就是:通过spring.factories加载的配置类优先级更低,我们自定义的装配类最后才会加载
2.2、@AutoConfigureXX注解失效原因总结
extendHandlerExceptionResolvers方法在WebMvcConfigurationSupport加载的时候已经执行过了,由于加载顺序问题,那么我们自己的装配类中重写方法,将无法被调用;所以这就是@AutoConfigureBefore、@AutoConfigureAfter和@AutoConfigureOrder注解无法生效的原因,不适用很多场景
3、使用静态内部装配类提升加载顺序
这也是小编使用的方法,这种场景在纯Spring环境下我们几乎遇不见,缘由是在Spring下所有的配置文件都是我们手动确定和编写,所以“哪些能写、哪些不能写,哪些在前,哪些在后”均是确定的,由我们程序员自行控制。该场景在Spring Boot场景下被大量用到
总结如下:
- static加在bean注册方法上
1、
@Configuration配置类最优先被初始化,才会继续初始化其里面的@Bean;若有多个 @Configuration配置类,顺序由你构造AnnotationConfigApplicationContext时传入的顺序为准(若是被scan扫描进去的,则无序)
2、 @Bean方法上加static成为静态方法,并不能提升此Bean的优先级
主要是因为@Bean的解析,必须是发生在@Configuration配置类被实例化后,因此它并不能提升优先级
- static加在静态内部类上
1、 @Configuration(外层)配置类的初始化顺序依旧是按照AnnotationConfigApplicationContext的定义顺序来的; 对于内部类的@Configuration的初始化(不管是静态还是非静态),也依旧是外部的@Configuration完成后才行
2、内部类里的@Bean的优先级均高于外层定义的@Bean,同时可以看到static静态内部类能够提升优先级,它比非静态内部类的优先级还高
3、内部类有限原则它只作用于本@Configuration类,也就是说仅在本主类内提升优先级。另外若出现多个内部类,按照定义顺序执行(static永远高于非static哦)
4、 内部类的访问权限无所谓,private都行。
4、bean加载顺序规则
直接上干货了,测试代码太长啦;
SpringBoot2.7以后自动装配的定义文件该成了org.springframework.boot.autoconfigure.AutoConfiguration.imports
- Spring.factories中定义的自动装配类优先级最低
- 本类中bean顺序按依赖优先+代码顺序原则,比如A/B/C三个Bean上中下顺序写的代码,A依赖了C,所以C比A和B都提前加载
- 跨类的bean加载顺序按照依赖优先+bean名称字母顺序加载
- 静态内部类bean与跨类bean加载顺序一致
相关文章:
SpringBoot自动装配定义先后顺序失效原因极其解析
SpringBoot自动装配定义先后顺序失效原因极其解析 1、场景分析1.1、问题总结 2、使用AutoConfigureBefore、AutoConfigureAfter和AutoConfigureOrder注解指定加载顺序2.2、AutoConfigureXX注解失效原因总结 3、使用静态内部装配类提升加载顺序4、bean加载顺序规则 1、场景分析 …...
API 集成测试工具Hitchhiker 0.1.1 正式发布
Hitchhiker 是一款开源的 Restful Api 集成测试工具,你可以在轻松部署到本地,和你的 team 成员一起管理 Api。 能做什么 * Team 协作开发 Api * Api 历史修改记录及支持 diff 展示 * 支持多环境变量及运行时变量 * 支持 Schedule 及批量 run * 不同…...
idea无法下载源码-Cannot download sources
问题: 解决方案:...
docker搭建mysql主从复制
1. 基础环境 环境 名称描述CentOS 7.6Linux操作系统版本docker 20.10.5docker版本mysql 8.0.29mysql镜像版本 节点 节点名称读写/主从地址端口master读节点/主节点192.168.1.6:3306slave1写节点/从节点192.168.1.6:3307slave2写节点/从节点192.168.1.6:3308 2. 主节点 使…...
在MacBook上实现免费的PDF文件编辑
之前我想对PDF文件进行简单处理(比如删页面、添空白页、调整页面顺序),要么是开wps会员【花钱贵】,下载(盗版)Adobe Acrobat【macOS不好下载】,要么用福昕阅览器登陆学生账号(学校买…...
QT第2课-GUI程序实例分析
GUI程序开发概述 不同的操作系统GUI开发原理相同不同的操作系统GUI SDK 不同 GUI 程序开发原理 GUI程序在运行时会创建一个消息队列系统内核将用户的键盘鼠标操作翻译成对应的程序消息程序在运行过程中需要实时处理队列中的消息当队列中没有消息时,程序将处于停滞…...
Android修行手册 - POI操作Excel常用样式(字体,背景,颜色,Style)
点击跳转>Unity3D特效百例点击跳转>案例项目实战源码点击跳转>游戏脚本-辅助自动化点击跳转>Android控件全解手册点击跳转>Scratch编程案例点击跳转>软考全系列 👉关于作者 专注于Android/Unity和各种游戏开发技巧,以及各种资源分享&…...
Sprint Boot 学习路线 5
Spring MVC Spring MVC是Spring框架的一部分,是一个Web应用程序框架。它旨在使用Model-View-Controller(MVC)设计模式轻松构建Web应用程序。 在Spring MVC中,应用程序被分为三个主要组件:Model、View和Controller。M…...
git02->gui图形化界面使用,ssh协议,idea集成GIT
gui图形化界面使用ssh协议idea集成GIT 1.gui图形化界面使用 2.ssh协议 git/github生成密钥并通过 操作分为本地电脑配置和github网站配置 第一步:本地电脑配置 右键空白处,选择Git Bash Here打开相关命令窗口 1.配置用户名和邮箱(如果已经配…...
面向对象 元类 gil log 协程 垃圾回收 描述符 property
封装、继承、多态 当谈到封装、继承和多态时,通常是在面向对象编程 (OOP) 的上下文中讨论的。 封装 (Encapsulation) 示例: class Person: def __init__(self, name, age): self.__name = name # 使用双下划线前缀将属性变为私有 self.__age = age de…...
bibitem格式 添加参考文献
这次写论文时遇到一种bibitem格式的参考文献,latex中没有bib文件 分三步走 找到这篇文章的Bib Tex的引用,然后新建bib文件,命名为下图: 然后把Bib Tex引用的内容复制到上图的文件中,新建tex文件 内容为 \document…...
Leetcode 2934. Minimum Operations to Maximize Last Elements in Arrays
Leetcode 2934. Minimum Operations to Maximize Last Elements in Arrays 1. 解题思路2. 代码实现 题目链接:2934. Minimum Operations to Maximize Last Elements in Arrays 1. 解题思路 这一题思路上其实很简单,直接分情况考察一下最后一个元素交换…...
02:2440---时钟体系
目录 一:时钟控制 1:基本概念 2:时钟结构图 3:结构图分析 4:总线 5:寄存器 A:FCLK--MPLLCON B:HCLK和PCLK--CLKDIVN C:注意 二:上电复位 1:上电复位 2:时钟选择 三:代码 一:时钟控制 1:基本概念 S3C2440A中的时钟控制逻辑可以产生所需的时钟信号,包括C…...
SOEM源码解析——ecx_siiSM(读取SII的SM信息)
0 工具准备 1.SOEM-master-1.4.0源码1 ecx_siiSM函数总览 /** Get SM data from SII SM section in slave EEPROM.从SII的SM段获取SM信息* @param[in] context = context struct 句柄* @param[in] slave = slave number 从站序号* @param[out] SM = first SM str…...
【见缝插针】射击类游戏-微信小程序项目开发流程详解
还记得小时候玩过的见缝插针游戏吗,比一比看谁插得针比较多,可有趣了,当然了,通过它可以训练自己的手速反应,以及射击水平,把握时机,得分越高就越有成就感,相信小朋友们会喜欢它的&a…...
flutter开发实战-TweenSequence实现动画序列
flutter开发实战-TweenSequence实现动画序列 一、TweenSequence TweenSequence是允许创建一个Animation由一系列补间动画来确定值,每个TweenSequenceItem都有定义在动画的持续时间的权重确定动画间隔。 TweenSequence 动画组类TweenSequenceItem 用来定义每一个动…...
Flowable 外部表单
内置表单需要在每个节点中去配置,当如果多个节点使用同一套表单属性就要配置多次比较麻烦,修改的时候也要修改多次,外部表单可以定义一次,然后其它节点都去引用同一个表单属性。 外部表单需要定义一个.form后缀的文件。 外部表单…...
[mysql]索引优化-2
目录 一、分页查询优化1.根据自增且连续的主键排序的分页查询2.根据非主键字段排序的分页查询 二、Join关联查询优化1.嵌套循环连接 Nested-Loop Join(NLJ) 算法2.基于块的嵌套循环连接 Block Nested-Loop Join(BNL)算法 三、count(*)查询优化1.查询mysql自己维护的总行数2.sho…...
数据分析实战 | 泊松回归——航班数据分析
目录 一、数据及分析对象 二、目的及分析任务 三、方法及工具 四、数据读入 五、数据理解 六、数据准备 七、模型训练 八、模型评价 一、数据及分析对象 CSV文件:o-ring-erosion-only.csv 数据集链接:https://download.csdn.net/download/m0_7…...
Fliki AI:让视频创作更简单、更高效
在当今的数字时代,视频已经成为人们获取信息和娱乐的重要方式。无论是企业宣传、教育培训还是个人创作,视频都发挥着越来越重要的作用。然而,视频制作是一项复杂的工作,需要掌握一定的技能和经验。这对于初学者或没有专业视频制作…...
**发散创新:用 Rust实现高性能光线追踪渲染器的核心逻辑与优化技巧**在现代图形学领域,**光线追踪
发散创新:用 Rust 实现高性能光线追踪渲染器的核心逻辑与优化技巧 在现代图形学领域,光线追踪(Ray Tracing) 已成为高质量实时渲染的基石。随着硬件支持(如 NVIDIA RTX)普及,越来越多开发者开始…...
密码学·顶级会议与资源导航
1. 密码学研究的黄金殿堂:三大顶级会议详解 第一次接触密码学领域时,最让我困惑的就是如何找到高质量的学术资源。直到导师告诉我:"盯住三大会议,你就抓住了密码学的命脉。"这句话彻底改变了我的研究方向。Crypto、Euro…...
从0到1打造完美PRD:这10个细节让你的需求文档更专业
从0到1打造完美PRD:这10个细节让你的需求文档更专业 在跨部门协作的产品开发中,一份优秀的PRD(产品需求文档)如同航海图,既能指引团队方向,又能规避潜在风险。但现实中,许多产品经理的文档常陷入…...
从零到一:手把手教你用Python玩转ISO14443读卡(附完整代码与调试技巧)
从零到一:手把手教你用Python玩转ISO14443读卡(附完整代码与调试技巧) 在物联网和嵌入式开发领域,与智能卡进行通信是一项基础但关键的技能。ISO14443作为近场通信(NFC)的主流标准之一,广泛应用…...
AI Coding越来越强,我们还有必要学Processing吗? · 创意编程陕
故障表现 发现请求集群 demo 入口时卡住,并且对应 Pod 没有新的日志输出 rootce-demo-1:~# kubectl get pods -n deepflow-otel-spring-demo -o wide NAME READY STATUS RESTARTS AGE IP NODE NOMINATED NO…...
深入解析C99中函数隐式声明无效警告的根源与解决方案
1. 为什么C99标准对函数隐式声明如此严格? 我第一次在嵌入式项目里遇到这个警告时,整个人都是懵的。当时正在调试STM32的定时器初始化代码,编译时突然蹦出"Warning: implicit declaration of function TIM2_Int_Init is invalid in C99&…...
信托资金流向与交易对手辨析:钱给了谁,谁就是交易对手吗?
目录 一、 核心误区:资金流向 ≠ 交易对手 二、 谁才是真正的“交易对手”? 三、 如何一眼识别真正的交易对手? 总结 在信托业务和资产管理领域,很多初学者甚至从业者容易产生一个误区:认为信托公司把钱打给谁&…...
OpenClaw内存优化技巧:Phi-3-vision-128k-instruct在8GB设备上的稳定运行方案
OpenClaw内存优化技巧:Phi-3-vision-128k-instruct在8GB设备上的稳定运行方案 1. 为什么需要内存优化? 去年我在一台老款MacBook Air上第一次尝试部署Phi-3-vision-128k-instruct时,系统几乎立即崩溃。这台仅有8GB内存的设备,在…...
linux驱动调试方法整理
一般我喜欢把linux驱动的调试按矛盾分为几大类:1.驱动工程师和内核/自己的 2.应用工程师和驱动工程师的。其中我们需要掌握linux众多的调试控件才能灵活运用。驱动工程师和内核/自己的1.驱动加载和驱动调试信息使用printk(per_log)打印需求使…...
从电子琴到智能家居:无源蜂鸣器如何玩出花样?附ESP32播放《超级玛丽》主题曲代码
无源蜂鸣器的创意交响曲:从电子琴到游戏音效的ESP32实战指南 在创客和物联网开发的世界里,声音交互常常是项目中最容易被忽视却又最能提升用户体验的环节。无源蜂鸣器这个看似简单的元件,实际上蕴藏着惊人的创意潜力——它不只是发出单调的&q…...
