文章目录

  • 问题描述
  • 解决方案
  • 问题分析
  • 到底allowBeanDefinitionOverriding应该设置true还是false?

问题描述

最近在学习spring cloud sleuth过程中,遇到了一个问题:

The bean 'characterEncodingFilter', defined in class path resource [zipkin/autoconfigure/ui/ZipkinUiAutoConfiguration.class], could not be registered. A bean with that name has already been defined in class path resource [org/springframework/boot/autoconfigure/web/servlet/HttpEncodingAutoConfiguration.class] and overriding is disabled.
Action:Consider renaming one of the beans or enabling overriding by setting spring.main.allow-bean-definition-overriding=true

从错误信息中可以看到,characterEncodingFilter这个bean被定义了两次,ZipkinUiAutoConfiguration和HttpEncodingAutoConfiguration都有定义。在大型项目开发过程中,这种情况并不少见。毕竟各个不同的组件都是独立开发的,集成到一起后总会遇到各种惊喜。spring.main
.allow-bean-definition-overriding=true就是解决bean重复定义的。设置为true时,后定义的bean会覆盖之前定义的相同名称的bean。

具体参见我的另一篇文章 spring cloud Greenwich 学习笔记(十)spring cloud sleuth
服务链路追踪。

解决方案

application.properties中配置,并且提示了我们要配置spring.main .allow-bean-definition-overriding=true

或者

application.yml中配置

spring:main:allow-bean-definition-overriding: true

问题分析

上面已经看到,spring.main.allow-bean-definition-overriding设置为true,表示后发现的bean会覆盖之前相同名称的bean。
问题就出在spring初始化时bean工厂加载bean的时候。我们来看一下DefaultListableBeanFactory代码:

  /** 是否允许使用相同名称重新注册不同的bean实现. 默认是允许*/private boolean allowBeanDefinitionOverriding = true;/*** Set whether it should be allowed to override bean definitions by registering* a different definition with the same name, automatically replacing the former.* If not, an exception will be thrown. This also applies to overriding aliases.* <p>Default is "true".【这里明确说明了默认是true】* @see #registerBeanDefinition*/public boolean isAllowBeanDefinitionOverriding() {return this.allowBeanDefinitionOverriding;}@Overridepublic void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)throws BeanDefinitionStoreException {Assert.hasText(beanName, "Bean name must not be empty");Assert.notNull(beanDefinition, "BeanDefinition must not be null");if (beanDefinition instanceof AbstractBeanDefinition) {try {((AbstractBeanDefinition) beanDefinition).validate();}catch (BeanDefinitionValidationException ex) {throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,"Validation of bean definition failed", ex);}}//bean加载到spring的工程中后,会存储在beanDefinitionMap中,key是bean的名称。BeanDefinition existingDefinition = this.beanDefinitionMap.get(beanName);if (existingDefinition != null) {//不为空,说明相同名称的bean已经存在了if (!isAllowBeanDefinitionOverriding()) {//如果不允许相同名称的bean存在,则直接抛出异常throw new BeanDefinitionOverrideException(beanName, beanDefinition, existingDefinition);}else if (existingDefinition.getRole() < beanDefinition.getRole()) {// e.g. was ROLE_APPLICATION, now overriding with ROLE_SUPPORT or ROLE_INFRASTRUCTUREif (logger.isInfoEnabled()) {logger.info("Overriding user-defined bean definition for bean '" + beanName +"' with a framework-generated bean definition: replacing [" +existingDefinition + "] with [" + beanDefinition + "]");}}else if (!beanDefinition.equals(existingDefinition)) {if (logger.isDebugEnabled()) {logger.debug("Overriding bean definition for bean '" + beanName +"' with a different definition: replacing [" + existingDefinition +"] with [" + beanDefinition + "]");}}else {if (logger.isTraceEnabled()) {logger.trace("Overriding bean definition for bean '" + beanName +"' with an equivalent definition: replacing [" + existingDefinition +"] with [" + beanDefinition + "]");}}//可见,上面allowBeanDefinitionOverriding =true时,只是记录了一些日志,然后后来发现的这个bean,会覆盖之前老的bean。this.beanDefinitionMap.put(beanName, beanDefinition);}else {if (hasBeanCreationStarted()) {// Cannot modify startup-time collection elements anymore (for stable iteration)synchronized (this.beanDefinitionMap) {this.beanDefinitionMap.put(beanName, beanDefinition);List<String> updatedDefinitions = new ArrayList<>(this.beanDefinitionNames.size() + 1);updatedDefinitions.addAll(this.beanDefinitionNames);updatedDefinitions.add(beanName);this.beanDefinitionNames = updatedDefinitions;if (this.manualSingletonNames.contains(beanName)) {Set<String> updatedSingletons = new LinkedHashSet<>(this.manualSingletonNames);updatedSingletons.remove(beanName);this.manualSingletonNames = updatedSingletons;}}}else {// Still in startup registration phasethis.beanDefinitionMap.put(beanName, beanDefinition);this.beanDefinitionNames.add(beanName);this.manualSingletonNames.remove(beanName);}this.frozenBeanDefinitionNames = null;}if (existingDefinition != null || containsSingleton(beanName)) {resetBeanDefinition(beanName);}}

具体可以看上面代码中添加的注释。
可以看一下BeanDefinitionOverrideException的定义,我们上面抛出的异常,就是这里抛出的。

public BeanDefinitionOverrideException(String beanName, BeanDefinition beanDefinition, BeanDefinition existingDefinition) {super(beanDefinition.getResourceDescription(), beanName,"Cannot register bean definition [" + beanDefinition + "] for bean '" + beanName +"': There is already [" + existingDefinition + "] bound.");this.beanDefinition = beanDefinition;this.existingDefinition = existingDefinition;
}

实际错误时提示bean重复,不过这里spring又对其进行了封装,最终打印出来的结果就是本文开头的错误输出,并且提示了我们要配置spring.main .allow-bean-definition-overriding=true

可能有的人会问了,上面代码中默认就是true啊,为什么还要我手动配置?

原因就是上面贴出来的是spring的代码,而springboot对这个参数又进行了二次封装,springboot中的allowBeanDefinitionOverriding是没有初始化默认值的,我们知道,java中的boolean类型不初始化时是false。
springboot中源代码:

//没有默认初始化就是false
private boolean allowBeanDefinitionOverriding;
/*** Sets if bean definition overriding, by registering a definition with the same name* as an existing definition, should be allowed. Defaults to {@code false}.【这里写的很明白了,默认是false】* @param allowBeanDefinitionOverriding if overriding is allowed* @since 2.1* @see DefaultListableBeanFactory#setAllowBeanDefinitionOverriding(boolean)*/
public void setAllowBeanDefinitionOverriding(boolean allowBeanDefinitionOverriding) {this.allowBeanDefinitionOverriding = allowBeanDefinitionOverriding;
}private void prepareContext(ConfigurableApplicationContext context,ConfigurableEnvironment environment, SpringApplicationRunListeners listeners,ApplicationArguments applicationArguments, Banner printedBanner) {context.setEnvironment(environment);postProcessApplicationContext(context);applyInitializers(context);listeners.contextPrepared(context);if (this.logStartupInfo) {logStartupInfo(context.getParent() == null);logStartupProfileInfo(context);}// Add boot specific singleton beansConfigurableListableBeanFactory beanFactory = context.getBeanFactory();beanFactory.registerSingleton("springApplicationArguments", applicationArguments);if (printedBanner != null) {beanFactory.registerSingleton("springBootBanner", printedBanner);}if (beanFactory instanceof DefaultListableBeanFactory) {//**在此处给bean工程设置属性**((DefaultListableBeanFactory) beanFactory).setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);}// Load the sourcesSet<Object> sources = getAllSources();Assert.notEmpty(sources, "Sources must not be empty");load(context, sources.toArray(new Object[0]));listeners.contextLoaded(context);
}

到底allowBeanDefinitionOverriding应该设置true还是false?

一句话:按需设置
上面代码中可以看到,spring中默认是true,也就是默认支持名称相同的bean的覆盖。而springboot中的默认值是false,也就是不支持名称相同的bean被覆盖。
那么我们自己应该如何选择呢?
这里笔者认为默认不覆盖比较好。
因为还是推荐一个系统中不要存在名称相同的bean,否则后者覆盖前者,多人分工合作的时候,难以避免某些bean被覆盖,会出现很多诡异的问题
,甚至会带来线上真实的业务损失。
bean的名称不相同,依据具体的业务给bean起名字。这样不但可以解决bean名称重复的问题,还可以大大提高程序的可读性与可维护性。
只有当集成了第三方的库,不同库直接由于是多个团队开发的,甚至这些团队属于不同的国家,有可能会出现bean名称相同的情况。这种情况就需要根据实际需求来设置allowBeanDefinitionOverriding的值了。

spring中 allowBeanDefinitionOverriding(spring.main.allow-bean-definition-overriding)原因分析、解决办法相关推荐

  1. JS中for循环里面的闭包问题的原因及解决办法

    JS中for循环里面的闭包问题的原因及解决办法 参考文章: (1)JS中for循环里面的闭包问题的原因及解决办法 (2)https://www.cnblogs.com/ZinCode/p/555190 ...

  2. spring中 allowBeanDefinitionOverriding(spring.main.allow-bean-definition-overriding) 分析

    文章目录 问题描述 问题分析 到底allowBeanDefinitionOverriding应该设置true还是false? 问题描述 最近在学习spring cloud sleuth过程中,遇到了一 ...

  3. spring中 allowBeanDefinitionOverriding(spring.main.allow-bean-definition-overriding) 分析

    文章目录 问题描述 问题分析 到底allowBeanDefinitionOverriding应该设置true还是false? 问题描述 最近在学习spring cloud sleuth过程中,遇到了一 ...

  4. CAD图纸中CAD文字边界为锯齿形的原因及解决办法

    在使用浩辰CAD软件绘制CAD图纸的过程中,为了方便绘图/看图经常需要在图纸中输入一些文字.那么当图纸中CAD文字边界显示为锯齿形时该怎么办呢?下面就给大家详细介绍一下吧! CAD文字边界为锯齿形的原 ...

  5. 【error】see previous definition of原因及解决办法

    报错:error: previous definition of 原因如下: 现在有A.h B.h C.h 三个头文件 其中 B.h中 #include<A.h> C.h中 #includ ...

  6. 浏览器打开服务器上的图片无法显示,网页中的图片打不开怎么办?原因与解决办法...

    最近有网友问小编这样一个很泛的问题:网页中的图片打不开怎么办?对于这个问题,其实导致的原因有很多,但也很好排除原因,主要从网络,网页,平台等当面综合去分析,就很容易可以找到答案.以下是小编对网页中的图 ...

  7. 棋牌APP下载链接在微信中被封拦截屏蔽无法打开的原因和解决办法

    接上一篇文章:https://blog.csdn.net/rolycn16/article/details/89014862 我们继续聊聊微信中浏览器下载链接打不开的情况 在我们做营销活动或推广宣传的 ...

  8. 彻底解决TortoiseGit中.gitignore文件失效,忽略文件失效原因及解决办法

    我更新了我的 .gitignore 文件,现在我想更新索引. 我想执行 git rm --cached 命令.但是我该怎么做TortoiseGit? 答案是: 右键单击该文件,选择TortoiseGi ...

  9. Eclipse工程中java文件上有小问号的原因和解决办法

    这是由于没有同步的原因造成的 ?的文件为服务器上没有的文件,及由你添加文件.需要上传至服务器或者commit一下. 也可以直接 对这工程->team->commit

  10. Spring中ref local与ref bean区别

    为什么80%的码农都做不了架构师?>>>    Spring中ref local与ref bean区别 今天在做SSH框架Demo实例时,在ApplicationResources. ...

最新文章

  1. SRWebSocket源码浅析(上)
  2. Java基础巩固——反射
  3. linux命令(常用)
  4. kotlin 查找id_Kotlin程序查找Square区域
  5. php和js序列化,PHP中serialize和json序列化与反序列化的区别
  6. 查看java运行时参数_查看JVM运行时参数
  7. php获取工作日时间,ThinkPHP中获取指定日期后工作日的具体日期方法
  8. iphone7无服务_iPhone 7 系列被召回?设计缺陷...
  9. oracle 数据分列,如何使用Excel把有规律地txt文本数据分列
  10. NYOJ.904 search(二分查找,快速排序,结构体运用)
  11. k380没有验证码_罗技K380蓝牙键盘
  12. 网易通行证html代码,JavaScript实现仿网易通行证表单验证
  13. java 汉字区位码表,中文汉字编码知识及各种中文编码对应的编码区间总结
  14. oracle数据库答辩,数据库课程设计答辩.ppt
  15. java学生成绩管理系统类图,学生成绩管理系统的用例类图.ppt
  16. c#中datagridview清空数据并删除空白行
  17. FSR402压力传感器数据转换
  18. 使用curl控制下载速度
  19. 「自动搬运+CDN」FFmpeg + x264 t_mod + x265 yuuki
  20. python数据分析与可视化从入门到精通_零基础学Python爬虫、数据分析与可视化从入门到精通...

热门文章

  1. [Linux]FloppyLinux--中国石油大学(华东)计算机操作系统课程设计作业教程
  2. 键盘计算机编程是什么意思,可编程键盘是什么
  3. 电视dns服务器修复,电信电视dns遭到劫持的解决方法分享
  4. _itemmod_extract_enchant
  5. 手把手带你调参Yolo v5 (v6.2)(训练)
  6. 多路耦合器(有源分离器)在无线通讯中的应用
  7. KEIL arm C51中常量变量函数的绝对地址设定
  8. wgc84 笛卡尔_WGS84椭球下的UTM坐标与Clarke80椭球下的兰勃特坐标转换方法研究
  9. vue导出word文档
  10. 网络信息安全:五、GRE和IPSEC