Spring動態(tài)替換Properties配置變量

Spring動態(tài)替換Properties配置變量

有如下需求,在Spring啟動的時候,我們需要修改某個指定的配置的值,從而達到動態(tài)加載變量的效果:

    @Value("${destination}")
    public String destination;

1.實現(xiàn)方法有如下:

  • 在對應(yīng)的Spring Bean實例化之后,通過反射動態(tài)的修改實例的參數(shù)值
  • 動態(tài)的修改PropertySource

具體的類如下:

public interface HelloService {

    /**
     * say hello
     */
    void sayHello();
}


@Service
public class HelloServiceImpl implements HelloService{

    @Value("${destination}")
    private String destination;

    @Override
    public void sayHello() {
        System.out.println("Hello, let us go to " + destination);
    }
}

第一種方法實現(xiàn)

方式1:
定義一個應(yīng)用事件,監(jiān)聽?wèi)?yīng)用上下文刷新事件,然后我們可以通過這個事件拿到ApplicationContext上下文,進而去獲取對應(yīng)的SpringBean,然后進行反射賦值操作。

public class LocalApplicationListener implements ApplicationListener<ContextRefreshedEvent> {

    @Override
    public void onApplicationEvent(ContextRefreshedEvent event) {
        Object helloService = event.getApplicationContext().getBean(HelloService.class);
        if (helloService instanceof HelloServiceImpl) {
            try {
                Field field = HelloServiceImpl.class.getDeclaredField("destination");
                field.setAccessible(true);
                ReflectionUtils.setField(field, helloService, "廣東深圳");
            } catch (NoSuchFieldException e) {
                e.printStackTrace();
            }
        }
    }
}

有了這個監(jiān)聽事件,我們需要把這個監(jiān)聽器添加到Spring的監(jiān)聽器集合中,可以通過啟動類的啟動方法進行添加:

@SpringBootApplication
public class DemoApplication {

    public static void main(String[] args) {
        SpringApplication app = new SpringApplication(DemoApplication.class);
        app.addListeners(new LocalApplicationListener());
        app.run(args);
    }
}

方式2:
定義一個類實現(xiàn)BeanPostProcessor接口,在接口中進行反射操作。BeanPostProcessor是Spring提供的一個Bean擴展接口,可以通過該接口實現(xiàn)一些Bean創(chuàng)建之前和創(chuàng)建之后的操作。

@Component
public class LocalBeanProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof HelloServiceImpl) {
            try {
                Field field = HelloServiceImpl.class.getDeclaredField("destination");
                field.setAccessible(true);
                ReflectionUtils.setField(field, bean, "廣東廣州");
            } catch (NoSuchFieldException e) {
                e.printStackTrace();
            }
        }
        return bean;
    }
}

第二種實現(xiàn)方法
通過了解Spring的加載流程,我們得知Spring在上下文準(zhǔn)備完畢(配置信息解析完畢并創(chuàng)建好了應(yīng)用上下文)之后,會調(diào)用推送一個ApplicationEnvironmentPreparedEvent,從這個事件中,我們可以獲取到ConfigurableEnvironment對象,而ConfigurableEnvironment對象可以獲取到MutablePropertySources。對于MutablePropertySources,它包含了Spring的所有 的配置信息,包括我們啟動應(yīng)用的application.properties文件的配置信息。具體代碼如下:

public class LocalEnvironmentPrepareEventListener implements ApplicationListener<ApplicationEnvironmentPreparedEvent> {

    @Override
    public void onApplicationEvent(ApplicationEnvironmentPreparedEvent event) {
        MutablePropertySources propertySources = event.getEnvironment().getPropertySources();
        for (PropertySource<?> propertySource : propertySources) {
            boolean applicationConfig = propertySource.getName().contains("applicationConfig");
            if (!applicationConfig) {
               continue;
            }
            Object property = propertySource.getProperty("destination");
            Map<String, OriginTrackedValue> source = (Map<String, OriginTrackedValue>) propertySource.getSource();
            OriginTrackedValue originTrackedValue = source.get("destination");
            OriginTrackedValue newOriginTrackedValue = OriginTrackedValue.of("中國香港", originTrackedValue.getOrigin());
            source.put("destination", newOriginTrackedValue);
            System.out.println(property);
        }
    }
}

同樣的我們需要把該監(jiān)聽器添加到Spring中:

@SpringBootApplication
public class DemoApplication {

    public static void main(String[] args) {
        SpringApplication app = new SpringApplication(DemoApplication.class);
        app.addListeners(new LocalEnvironmentPrepareEventListener());
        app.run(args);
    }
}

知名的分布式配置管理工具Apollo就是通過反射來實現(xiàn)配置參數(shù)的動態(tài)修改的,Apollo實現(xiàn)了BeanPostProcessor接口,這樣它就可以把所有的Bean和@Value的key的關(guān)系保存起來,類似于Map<String, List<Bean>>,當(dāng)配置中心的配置被改動的時候,就發(fā)一個通知給對應(yīng)的服務(wù),然后由服務(wù)自己去拉取配置參數(shù),重新賦值。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 本文是我自己在秋招復(fù)習(xí)時的讀書筆記,整理的知識點,也是為了防止忘記,尊重勞動成果,轉(zhuǎn)載注明出處哦!如果你也喜歡,那...
    波波波先森閱讀 12,440評論 6 86
  • 本來是準(zhǔn)備看一看Spring源碼的。然后在知乎上看到來一個帖子,說有一群**自己連Spring官方文檔都沒有完全讀...
    此魚不得水閱讀 7,036評論 4 21
  • Spring容器高層視圖 Spring 啟動時讀取應(yīng)用程序提供的Bean配置信息,并在Spring容器中生成一份相...
    Theriseof閱讀 2,914評論 1 24
  • 商業(yè)的競爭不僅是對外的行為也可內(nèi)部的競爭。 外部競爭更多的是利益關(guān)系,用不同的經(jīng)營模式擴展事...
    楊平的閱讀 136評論 0 0
  • 晚上在群里聊天的時候,對照價值感刻度表,發(fā)現(xiàn)自己在價值感2度,老師說進群的都是零度,我就困惑了。 ...
    o糖果罐o閱讀 316評論 0 1

友情鏈接更多精彩內(nèi)容