Bug:引入Feign后触发了2次、4次ContextRefreshedEvent

Bug:引入Feign后发现监控onApplication中ContextRefreshedEvent事件触发了2次或者4次。
【原理】在Spring的文档注释中提示到: Event raised when an {@code ApplicationContext} gets initialized or refreshed.即当 ApplicationContext 进行初始化或者刷新时都会发送该事件。只有 finishRefresh() 方法里面会发送该事件,而该方法又只有 refresh() 方法调用,在 refresh() 方法中有这些注释: Load or refresh the persistent representation of the configuration, which ight be from Java-based configuration, an XML file, a properties file, a elational database schema, or some other format. As this is a startup method, it should destroy already created singletons if it fails, to avoid dangling resources. In other words, after invocation of this method, either all or no singletons at all should be instantiated.加载或者刷新配置文件,由于这是一个启动方法,如果失败,它应该销毁已经创建的单例,以避免悬空资源。换句话说,在调用这个方法前,要么全部实例化,要么根本不实例化。在初始化 openFeign 组件的最后,会调用 SubContext 的 refresh()操作,最终会触发 SubContext 发出ContextRefreshedEvent事件。
为啥出现多次触发ContextRefreshedEvent **】监控类
**的每个ApplicationContext都会加载两次。在 Spring 框架中,事件(Event)是会沿着 Context 层次向上传播。简单来说子 Context 发出初始化完成事件,进而引发父 Context 也发出相同事件,而父 Context 此时并没有真正初始化完成。

  • AnnotationConfigApplicationContext~A
  • AnnotationConfigServletWebServerApplicationContext~B
  • AnnotationConfigApplicationContext~C

监控类#1****每个ApplicationContext触发4次ContextRefreshEvent,下为父子关系:

  • A:bootstrap:@21947
    • B:fkgss-serviceproxy-1:@17804
      • C:FeignContext-dc-inboundaa-rooladc:@17795,触发2次
    • B**:fkgss-serviceproxy-1:@17804,触发2次**

监控类#2****只触发2次ContextRefreshEvent。

  • A:bootstrap:@21932
    • B:fkgss-serviceproxy-1:@17807
      • C:FeignContext-dc-inboundaa-rooladc:@17799,触发1次
    • B:fkgss-serviceproxy-1:@17807,触发1次

**【验证】新增一个Feign,会出现三次,验证是父子上下文导致。**监控类#2问题解决。

  • 空白项目接入监控类#1模块,验证是否因为外部SDK原因导致:触发1次。
  • 父子项目接入监控类#1:触发1次。
  • 接入OpenFeign的同时接入监控类#1:触发1次。

=========>XX项目问题<=========

可能因为kgs中引入多个监控类#1:检索后没有发现。打印Thread.dumpStack()后:

  • 在第一个堆栈中,onApplicationEvent 方法位于监控类#1类中,该方法是在 SimpleApplicationEventMulticaster 发布的事件中触发的。该事件通常在 finishRefresh 或 refresh 方法调用时触发,这些方法是在 Spring Boot 应用程序启动时执行的。
  • 在第二个堆栈中,onApplicationEvent 方法被通过 CGLIB 动态代理调用。CGLIB 代理的存在通常意味着 SeqNo 类被 Spring AOP 代理了,因此通过代理调用的 onApplicationEvent 可能触发了额外的逻辑,如切面增强、方法拦截等。

【触发条件】

  • Spring Event Multicasting:SimpleApplicationEventMulticaster 会为每个事件调用所有的事件监听器(如 SeqNo.onApplicationEvent)。因此,每当 ContextRefresh 事件发布时,监听器(如 SeqNo)会被触发一次。
  • CGLIB 代理的影响:由于 SeqNo 类可能是被 AOP 代理的,因此代理类的调用可能会导致同一方法被调用多次。具体来说,CGLIB 代理会生成一个新的类,并将对 SeqNo 方法的调用拦截和代理。这种代理逻辑可能导致事件监听方法被多次调用,导致 onApplicationEvent 被触发两次。

验证监控类#1是否是代理类

搜索日志发现:Bean '监控类#1' of type [******] is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying);通常是由于 Spring 容器中的 Bean 在创建和初始化的过程中未能完全处理,导致该 Bean 无法被所有的 BeanPostProcessor 处理,尤其是无法进行自动代理。正是因为Feign提前触发ContextRefresh 。这也是为啥没办法通过BeanPostProcessor#postProcessAfterInitialization判断监控类#1是否被代理的原因。

【最后校验】

于是等【主上下文】加载完后,确定监控类#1被代理: 代理对象和原始 Bean 双重监听事件: 当事件发布时,Spring 容器会调用所有监听器的 onApplicationEvent 方法。由于 AOP 代理创建了一个新的 Bean 实例,该代理实例也会处理事件。因此,事件会被触发两次:

  • 一次是原始的 Bean 实例(即没有代理的实例)。
  • 另一次是代理实例。

**事件监听机制:**Spring 的事件监听机制是基于 Bean 的类型而不是 Bean 的实际实例。如果该 Bean 被 AOP 代理了,代理对象依然会注册为一个事件监听器,从而导致事件会被处理两次。

相关推荐
laocooon52385788610 小时前
写代码 调bug相关信息
bug
鸿儒51710 小时前
记录一个C++操作8位影像的一个bug
开发语言·c++·bug
·云扬·13 小时前
【Bug】 Elasticsearch启动失败(exit code 78):2个bootstrap检查问题排查与解决
elasticsearch·bootstrap·bug
BrianGriffin1 天前
get_download_file_path: command not found (asdf bug已解决)
bug
咸虾米_1 天前
解决九两酒商城[uni-pay-co]: Error:执行失败,积 分需要大于等于1的bug
bug·unicloud云开发·微信小程序商城·uniapp项目
凯子坚持 c1 天前
Git分支实战指南:如何优雅地管理版本、修复Bug并解决合并冲突
git·bug
北冥有渔jy1 天前
BT6.0常见的BUG
网络·安全·bug·蓝牙
测试19982 天前
一个只能通过压测发现Bug
自动化测试·软件测试·python·selenium·测试工具·bug·压力测试
狂奔的sherry2 天前
网卡获取模组ip失败问题解析
bug
nnsix3 天前
Unity ReferenceFinder插件 多选资源查找bug解决
unity·游戏引擎·bug