当心!不要在SpringBoot中再犯这样严重的错误

news/2024/10/5 9:07:07/

1. 简介

在Spring Boot中,@Configuration注解用于声明配置类,以定义和注册Bean对象。这些Bean对象可以是普通的业务组件,也可以是特殊的处理器,如BeanPostProcessor或BeanFactoryPostProcessor,用于在Spring容器中对其他Bean进行额外的处理。接下来我们将详细的介绍关于在SpringBoot环境下各种不正确的配置导致的各种问题。

2. 实战案例

2.1 循环依赖错误

当我们在一个配置类中使用@PostConstruct注解并且在其方法内部去引用其它Bean时,将会出现循环依赖错误,如下示例:

复制

@Configuration
public class AppConfig {@PostConstructpublic void init() {dao() ;System.out.println("AppConfig init...") ;}@BeanDAO dao() {return new DAO() ;}
}

在init()方法中调用dao()方法后,将无正确的启动SpringBoot,抛出如下错误

图片

图片

循环依赖错误,导致该错误的原因是非静态@Bean方法在语义上需要一个完全初始化的配置类实例来调用;简单点说就是在调用dao方法时需要完全的初始化AppConfig类,但是@PostConstruct注解的方法在执行时当前的这个AppConfig并没有完全的执行完成。要解决该问题可以通过如下2种方式:

方式1:

开启循环依赖

复制

spring:main:allow-circular-references: true

从SpringBoot2.6+开始默认不允许循环依赖。这样SpringBoot程序就能正确启动,不过这不是最好的方式也不推荐该种方式。

方式2:

将上面的dao方法声明为static方法;

复制

@Bean
public static DAO dao() {return new DAO() ;
}

static修饰的方法不需要包裹它的配置类提起初始化完成。这也是最为推荐的方法。

2.2 自定义处理器错误

当通过 @Bean 定义 BeanPostProcessor 和 BeanFactoryPostProcessor 时可能导致当前配置依赖注入的bean将不会生效(也就是@Autowired和@Value注解可能没有生效),如下示例:

复制

@Configuration
public class AppConfig {@Value("${pack.title}")private String title ;@Overridepublic String toString() {return "AppConfig [title=" + title + "]";}
}

配置文件中配置信息;

复制

pack:title: xxxooo

控制台输出

复制

AppConfig [title=xxxooo]

没有问题;但是如果你在AppConfig配置类中注册BeanPostProcessor后会出现什么情况呢?

自定义BeanPostProcessor;

复制

public class PackBeanPostProcessor implements BeanPostProcessor {// TODO
}

通过@Bean注册上面的BeanPostProcessor;

复制

@Bean
public PackBeanPostProcessor packBeanPostProcessor() {return new PackBeanPostProcessor() ;
}

再次运行服务,控制台输出

复制

AppConfig [title=xxxooo]

还是能正确的输出!?注意接下来我们对上面的自定义处理器做如下修改;

复制

public class PackBeanPostProcessor implements BeanPostProcessor, PriorityOrdered {// TODO@Overridepublic int getOrder() {return -1 ;}
}

这时候我们去实现了PriorityOrdered优先级接口,并将优先级设置的比较的高。如上调整后再次启动服务

复制

AppConfig [title=null]

问题出现了配置的属性并没有正确的解析注入,这是因为在默认情况下处理@Value注解的处理器的优先级低于你当前自定义处理器的优先级,所以这就导致了问题。同样的如果你使用@Autowired或@Resource也将会导致问题,如下示例:

复制

@Configuration
public class AppConfig {@Resourceprivate Person person ;
}

输出结果:

复制

AppConfig [persnotallow=null]

同样不能被注入;

要解决该问题可以通过如下2种方式:

方式1:

通过实现ApplicationContextInitializer接口;

复制

public class PackApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {@Overridepublic void initialize(ConfigurableApplicationContext context) {context.getBeanFactory().addBeanPostProcessor(new PackBeanPostProcessor());}
}

注册该实现;

复制

org.springframework.context.Applicatinotallow=\
com.pack.PackApplicationContextInitializer

这种方式实现非常麻烦;推荐下面的第二种方式

方式2:

将@Bean对应的方法声明为static即可。

复制

@Bean
public static PackBeanPostProcessor packBeanPostProcessor() {return new PackBeanPostProcessor() ;
}

将该方法声明为static后,那么容器在获取BeanPostProcessor是不需要先实例化包裹它的类的实例。

其实对于@Configuration注解的配置类,如果你有需要注入的对象,官方建议采用参数的方式注入,如下示例:

复制

@Configuration
public class AppConfig {private final Person person ;public AppConfig(Person person) {this.person = person ;}
}

构造函数注入也是在任何形式下的推荐注入方式。

作为程序员,持续学习和充电非常重要,作为开发者,我们需要保持好奇心和学习热情,不断探索新的技术,只有这样,我们才能在这个快速发展的时代中立于不败之地。低代码也是一个值得我们深入探索的领域,让我们拭目以待,它将给前端世界带来怎样的变革。

介绍一款程序员都应该知道的软件JNPF快速开发平台,很多人都尝试用过它,它是功能的集大成者,任何信息化系统都可以基于它开发出来。

JNPF 可以实现应用从创建、配置、开发、测试到发布、运维、升级等完整生命周期的管理。减少了传统应用程序的代码编写量,通过图形化、可视化的界面,以拖放组件的方式,即可快速生成应用程序的产品,大幅降低了开发企业管理类软件的难度。


http://www.ppmy.cn/news/1473386.html

相关文章

数据库的存储过程

数据库的存储过程 文章目录 数据库的存储过程一、定义二、最简单的存储过程&#xff08;无传入参数无返回参数的存储过程&#xff09;三、需求一四、需求二五、需求三六、删除存储过程 一、定义 存储过程是一种数据库对象&#xff0c;它封装了一组 SQL 语句和逻辑操作&#xf…

C++修饰符类型

一、存储类运算符 auto&#xff08;自动存储类&#xff0c;但在现代C中&#xff0c;它通常用于自动类型推导&#xff09; register&#xff08;建议编译器将变量存储在寄存器中&#xff0c;但现代编译器通常忽略此关键字&#xff09; static&#xff08;静态存储类&#xff…

【ARMv8/v9 GIC 系列 5.2 -- GIC 分组介绍:Group 0 |Group 1| Non-Secure Group 1】

请阅读【ARM GICv3/v4 实战学习 】 文章目录 GIC Interrupt grouping中断分组配置寄存器GIC 中断分组介绍Group 0(安全组0)Group 1(安全组1)Non-Secure Group 1(非安全组1)总结及例子GIC Interrupt grouping ARM GICv3 通过中断分组机制,与ARMv8异常模型和安全模型进行…

论文学习——使用基于多项式拟合的预测算法求解动态多目标问题

论文题目&#xff1a;Solving dynamic multi-objective problems using polynomial fitting-based prediction algorithm 使用基于多项式拟合的预测算法求解动态多目标问题&#xff08;Qingyang Zhang , Xiangyu He,Shengxiang Yang , Yongquan Dong , Hui Song , Shouyong Ji…

vue 代理

前端请求数据&#xff0c;要用ajax&#xff0c;由于浏览器的同源策略&#xff0c;涉及到跨域&#xff0c;解决跨域之后才能访问后端接口 一、常用的发送一个ajax请求&#xff1a; 1、xhr new XMLHttpRequest(),真正开发中不常用 2、jq&#xff0c;jq主要功能是获取dom&#xff…

css_23_多列布局

常用值&#xff1a; column-count&#xff1a;指定列数&#xff0c;值是数字。 column-width&#xff1a;指定列宽&#xff0c;值是长度。 columns &#xff1a;同时指定列宽和列数&#xff0c;复合属性&#xff1b;值没有数量和顺序要求。 column-gap&#xff1a;设置列边距…

el-table封装点击列筛选行数据功能,支持筛选,搜索,排序功能

数据少的话&#xff0c;可以前端实现&#xff0c;如果多的话&#xff0c;建议还是请求接口比较合理父组件&#xff1a; <template> <div class"home"> <!-- <img alt"Vue logo" src"../assets/logo.png"> <HelloWorld …

MATLAB-SCSO-CNN-SVM,基于SCSO沙猫群算法优化卷积神经网络CNN结合支持向量机SVM数据分类(多特征输入多分类)

SCSO-CNN-SVM&#xff0c;基于SCSO沙猫群算法优化卷积神经网络CNN结合支持向量机SVM数据分类(多特征输入多分类) 1.数据均为Excel数据&#xff0c;直接替换数据就可以运行程序。 2.所有程序都经过验证&#xff0c;保证程序可以运行。 3.具有良好的编程习惯&#xff0c;程序均…