@RefreshScope那些事

@RefreshScope那些事

要说清楚RefreshScope,先要了解Scope

  • Scope(org.springframework.beans.factory.config.Scope)是Spring 2.0开始就有的核心的概念

  • RefreshScope(org.springframework.cloud.context.scope.refresh)是spring cloud提供的一种特殊的scope实现,用来实现配置、实例热加载。

  • Scope -> GenericScope -> RefreshScope


    scope_hierarchy.jpeg
  • Scope与ApplicationContext生命周期

    • AbstractBeanFactory#doGetBean创建Bean实例
     protected <T> T doGetBean(...){
        final RootBeanDefinition mbd = ...
        if (mbd.isSingleton()) {
            ...
        } else if (mbd.isPrototype())
           ...
        } else {
              String scopeName = mbd.getScope();
              final Scope scope = this.scopes.get(scopeName);
              Object scopedInstance = scope.get(beanName, new ObjectFactory<Object>() {...});
              ...
        }
        ...
     }
    
    • Singleton和Prototype是硬编码的,并不是Scope子类。 Scope实际上是自定义扩展的接口
    • Scope Bean实例交由Scope自己创建,例如SessionScope是从Session中获取实例的,ThreadScope是从ThreadLocal中获取的,而RefreshScope是在内建缓存中获取的。
  • @Scope 对象的实例化

    • @RefreshScope 是scopeName="refresh"的 @Scope
     ...
      @Scope("refresh")
      public @interface RefreshScope {
          ...
      }
    
    • @Scope 的注册 AnnotatedBeanDefinitionReader#registerBean
      public void registerBean(...){
        ...
        ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
          abd.setScope(scopeMetadata.getScopeName());
        ...
          definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
      }
    
    • 读取@Scope元数据, AnnotationScopeMetadataResolver#resolveScopeMetadata
    public ScopeMetadata resolveScopeMetadata(BeanDefinition definition) {
              AnnotationAttributes attributes = AnnotationConfigUtils.attributesFor(
                      annDef.getMetadata(), Scope.class);
              if (attributes != null) {
                  metadata.setScopeName(attributes.getString("value"));
                  ScopedProxyMode proxyMode = attributes.getEnum("proxyMode");
                  if (proxyMode == null || proxyMode == ScopedProxyMode.DEFAULT) {
                      proxyMode = this.defaultProxyMode;
                  }
                  metadata.setScopedProxyMode(proxyMode);
              }
    }
    
    • Scope实例对象通过ScopedProxyFactoryBean创建,其中通过AOP使其实现ScopedObject接口,这里不再展开

现在来说说RefreshScope是如何实现配置和实例刷新的

  • RefreshScope注册

    • RefreshAutoConfiguration#RefreshScopeConfiguration
      @Component
      @ConditionalOnMissingBean(RefreshScope.class)
      protected static class RefreshScopeConfiguration implements BeanDefinitionRegistryPostProcessor{
      ...
          registry.registerBeanDefinition("refreshScope",
          BeanDefinitionBuilder.genericBeanDefinition(RefreshScope.class)
                              .setRole(BeanDefinition.ROLE_INFRASTRUCTURE)
                              .getBeanDefinition());
      ...
      }
    
    • RefreshScope extends GenericScope, 大部分逻辑在 GenericScope 中
    • GenericScope#postProcessBeanFactory 中向AbstractBeanFactory注册自己
    public class GenericScope implements Scope, BeanFactoryPostProcessor...{
          @Override
          public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory)
              throws BeansException {
              beanFactory.registerScope(this.name/*refresh*/, this/*RefreshScope*/);
              ...
          }
    }
    
  • RefreshScope 刷新过程

    • 入口在ContextRefresher#refresh
     refresh() {
          Map<String, Object> before = ①extract(
                  this.context.getEnvironment().getPropertySources());
          ②addConfigFilesToEnvironment();
          Set<String> keys = ④changes(before,
                  ③extract(this.context.getEnvironment().getPropertySources())).keySet();
          this.context.⑤publishEvent(new EnvironmentChangeEvent(keys));
          this.scope.⑥refreshAll();
     }
    
    • ①提取标准参数(SYSTEM,JNDI,SERVLET)之外所有参数变量
    • ②把原来的Environment里的参数放到一个新建的Spring Context容器下重新加载,完事之后关闭新容器
    • ③提起更新过的参数(排除标准参数)
    • ④比较出变更项
    • ⑤发布环境变更事件,接收:EnvironmentChangeListener/LoggingRebinder
    • ⑥RefreshScope用新的环境参数重新生成Bean
    • 重新生成的过程很简单,清除refreshscope缓存幷销毁Bean,下次就会重新从BeanFactory获取一个新的实例(该实例使用新的配置)
    • RefreshScope#refreshAll
      public void refreshAll() {
              <b>super.destroy();</b>
              this.context.publishEvent(new RefreshScopeRefreshedEvent());
      }
    
    • GenericScope#destroy
      public void destroy() {
          ...
          Collection<BeanLifecycleWrapper> wrappers = <b>this.cache.clear()</b>;
          for (BeanLifecycleWrapper wrapper : wrappers) {
              <b>wrapper.destroy();</b>
          }
      }
    
  • Spring Cloud Bus 如何触发 Refresh

    • BusAutoConfiguration#BusRefreshConfiguration 发布一个RefreshBusEndpoint
    @Configuration
      @ConditionalOnClass({ Endpoint.class, RefreshScope.class })
      protected static class BusRefreshConfiguration {
    
          @Configuration
          @ConditionalOnBean(ContextRefresher.class)
          @ConditionalOnProperty(value = "endpoints.spring.cloud.bus.refresh.enabled", matchIfMissing = true)
          protected static class BusRefreshEndpointConfiguration {
              @Bean
              public RefreshBusEndpoint refreshBusEndpoint(ApplicationContext context,
                      BusProperties bus) {
                  return new RefreshBusEndpoint(context, bus.getId());
              }
          }
      }
    
    • RefreshBusEndpoint 会从http端口触发广播RefreshRemoteApplicationEvent事件
     @Endpoint(id = "bus-refresh")
      public class RefreshBusEndpoint extends AbstractBusEndpoint {
           public void busRefresh() {
              publish(new RefreshRemoteApplicationEvent(this, getInstanceId(), null));
          }
      }
    
    • BusAutoConfiguration#refreshListener 负责接收事件(所有配置bus的节点)
      @Bean
      @ConditionalOnProperty(value = "spring.cloud.bus.refresh.enabled", matchIfMissing = true)
      @ConditionalOnBean(ContextRefresher.class)
      public RefreshListener refreshListener(ContextRefresher contextRefresher) {
          return new RefreshListener(contextRefresher);
      }
    
    • RefreshListener#onApplicationEvent 触发 ContextRefresher
    public void onApplicationEvent(RefreshRemoteApplicationEvent event) {
          Set<String> keys = contextRefresher.refresh();
      }
    
  • 大部分需要更新的服务需要打上@RefreshScope, EurekaClient是如何配置更新的

    • EurekaClientAutoConfiguration#RefreshableEurekaClientConfiguration
      @Configuration
      @ConditionalOnRefreshScope
      protected static class RefreshableEurekaClientConfiguration{
          @Bean
          @RefreshScope
          public EurekaClient eurekaClient(...) {
              return new CloudEurekaClient(manager, config, this.optionalArgs,
                      this.context);
          }
          
          @Bean
          @RefreshScope
          public ApplicationInfoManager eurekaApplicationInfoManager(...) {
              ...
              return new ApplicationInfoManager(config, instanceInfo);
          }
      }
    
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容