SpringCloud+@RefreshScope+@Value配置中心动态加载(刷新)配置文件

Posted OkidoGreen

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了SpringCloud+@RefreshScope+@Value配置中心动态加载(刷新)配置文件相关的知识,希望对你有一定的参考价值。

前言

现在的分布式系统中,通常会使用一个中间件作为配置中心,目的是实现在多服务器集群的场景下,只需要统一修改配置中心的配置文件,然后所有的服务器全部获取配置中心最新的配置文件信息,服务器不用重新启动,即可以在所有的服务器上面生效。要想实现这个功能,它的原理大概如下:

场景1:服务器首次启动-->获取配置中心最新的配置-->然后实例化bean,并根据配置文件信息初始化bean的所有的属性。这个场景很好理解,那些原本配置在本地的配置文件,也就是这样使用的,只不过配置中心的配置,需要远程获取一下。

场景2:配置中心配置信息修改->通知服务器获取最新的配置-->获取到最新的配置后,销毁所有的bean-->重新实例化所有的bean,然后根据最新的配置文件信息初始化所有的bean属性。这个过程就是SpringCloud框架,给我们实现的配置中心动态加载(刷新)配置文件。

注:下面所说的Spring框架包含:Spring FrameWork,Spring Boot,Spring Cloud。

Spring框架实现

在Spring框架中是这样实现的。

1.在类上加@RefreshScope注解。

2.在属性中引入@Value注解

3.手动调用刷新
http://localhost:8080/refresh,我猜想这个接口会触发销毁所有注解了@RefreshScope注解的bean,然后再后续使用bean的时候在实例化,初始化。因为没有看源码,我猜想是这个原理。

新版本中需要引入actuator,并添加如下的配置,来暴露refresh接口。

 

源码分析

来看一下调用过程。

@Endpoint(id = "refresh")
public class RefreshEndpoint 
   private ContextRefresher contextRefresher;   
   public RefreshEndpoint(ContextRefresher contextRefresher) 
      this.contextRefresher = contextRefresher;  
   
   @WriteOperation   
    public Collection<String> refresh() 
      //refresh方法最终会调用此逻辑
     Set<String> keys = this.contextRefresher.refresh(); //实际调用ContextRefresher去做的刷新逻辑    
      return keys;  
    

 
 
//this.contextRefresher.refresh();方法
public synchronized Set<String> refresh() 
		Set<String> keys = refreshEnvironment();
		this.scope.refreshAll();//实际调用这个方法去刷新
		return keys;
	
 
 
//this.scope.refreshAll();方法
@ManagedOperation(description = "Dispose of the current instance of all beans "
			+ "in this scope and force a refresh on next method execution.")
	public void refreshAll() 
		super.destroy();//销毁所有的bean
		this.context.publishEvent(new RefreshScopeRefreshedEvent());//发布刷新事件
	

我自己的疑惑是:

我用zookeeper作为配置中心,我发现我在zookeeper上面修改了配置信息,在应用程序中自动就生效了,因为没有找到源码,我猜想是Spring框架最终底层是调用了refreshAll()方法,但是我发现对象是没有被重新实例化的,因为通过IDE调试时,对象的编码还是和以前一样,如下图。

带着这个疑惑,继续看源码中...

后续:

1.当修改zookeeper配置中心的配置时,RefreshEventListener监听器会被触发,部分 源码如下:

private ContextRefresher refresh;
public void handle(RefreshEvent event) 
   if (this.ready.get())  // don't handle events before app is ready     
     log.debug("Event received " + event.getEventDesc());     
     Set<String> keys = this.refresh.refresh();  //实际调用ContextRefresher去做的刷新逻辑,
     //与手动调用   最终触发的逻辑是一样的。 
     log.info("Refresh keys changed: " + keys);   

在ContextRefresher方法中会触发配置信息刷新的逻辑,部分源码如下:

public synchronized Set<String> refreshEnvironment() 
   Map<String, Object> before = extract(
         this.context.getEnvironment().getPropertySources());   
        addConfigFilesToEnvironment();  //这个逻辑会触发重新去zookeeper获取最新的配置信息
       Set<String> keys = changes(before,  extract(this.context.getEnvironment().getPropertySources())).keySet();   
  this.context.publishEvent(new EnvironmentChangeEvent(this.context, keys));   
  return keys;//返回值是value值改变了的key集合

以上是关于SpringCloud+@RefreshScope+@Value配置中心动态加载(刷新)配置文件的主要内容,如果未能解决你的问题,请参考以下文章

SpringCloud+@RefreshScope+@Value配置中心动态加载(刷新)配置文件

SpringCloud+@RefreshScope+@Value配置中心动态加载(刷新)配置文件

SpringCloud Config手动刷新及自动刷新

检测 RefreshScope bean 的刷新

Spring Cloud @RefreshScope 原理分析:代理类调用流程

Spring Cloud @RefreshScope 原理分析:扫描 Bean 定义