Das Apollo
-Konfigurationscenter wurde im vorherigen Projekt verwendet. Nach dem Andocken des Apollo-Konfigurationscenters können die Eigenschaften des Konfigurationscenters im Programm verwendet werden Ist das Wolltuch umgesetzt? Wann wurden die Eigenschaften des Konfigurationscenters in das Programm geladen? Wenn wir also herausfinden, wie dies implementiert wird, können wir dann Konfigurationseigenschaften von überall laden
und Funktionen von Konfigurationseigenschaften verschlüsseln und entschlüsseln
? Apollo
配置中心,对接Apollo配置中心后,配置中心的属性就可以在程序中使用了,那么这个是怎么实现的呢?配置中心的属性又是何时加载到程序中的呢?那么我们如果找到了这个是怎么实现的是否就可以 从任何地方加载配置属性
、配置属性的加解密功能呢
?
从上图中得知,我们的需求很简单,即我们自己定义的属性需要比配置文件中的优先级更高。
当我们想在Bean中使用配置属性时,那么我们的配置属性必须在Bean实例化之前就放入到Spring到Environment中。即我们的接口需要在 application context refreshed
之前进行调用,而 EnvironmentPostProcessor
正好可以实现这个功能。
我们知道在 Spring中获取属性是有优先级的。
比如我们存在如下配置属性 username
├─application.properties │ >> username=huan ├─application-dev.properties │ >> username=huan.fu
那么此时 username
的值是什么呢?此处借用 Apollo
的一张图来说解释一下这个问题。
参考链接:https://www.apolloconfig.com/#/zh/design/apollo-design
Spring从3.1版本开始增加了ConfigurableEnvironment
和PropertySource
:
ConfigurableEnvironment
Spring的ApplicationContext会包含一个Environment(实现ConfigurableEnvironment接口)
ConfigurableEnvironment自身包含了很多个PropertySource
PropertySource
属性源
可以理解为很多个Key - Value的属性配置
由上方的原理图可知,key
在最开始出现的PropertySource
中的优先级更高,上面的例子在SpringBoot
中username
的值为huan.fu
。
由第二步 获取配置属性的优先级
可知,PropertySource
越靠前越先执行,那么要我们配置生效,就必须放在越前面越好。
由上图可知,SpringBoot加载各种配置是通过EnvironmentPostProcessor
来实现的,而具体的实现是ConfigDataEnvironmentPostProcessor
来实现的。那么我们自己编写一个EnvironmentPostProcessor
的实现类,然后在ConfigDataEnvironmentPostProcessor
后执行,并加入到 Environment
中的第一位即可。
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.6.6</version> <relativePath/> <!-- lookup parent from repository --> </parent> <groupId>com.huan.springcloud</groupId> <artifactId>springboot-extension-point</artifactId> <version>0.0.1-SNAPSHOT</version> <name>springboot-extension-point</name> <properties> <java.version>1.8</java.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> </project>
vim application.properties
username=huan
注意:
1、如果发现程序中日志没有输出,检查是否使用了slf4j
输出日志,此时因为日志系统未初始化无法输出日志。解决方法如下:
SpringBoot版本 >= 2.4 可以参考上图中的使用 DeferredLogFactory 来输出日志 < 2.4 1、参考如下链接 https://stackoverflow.com/questions/42839798/how-to-log-errors-in-a-environmentpostprocessor-execution 2、核心代码: @Component public class MyEnvironmentPostProcessor implements EnvironmentPostProcessor, ApplicationListener<ApplicationEvent> { private static final DeferredLog log = new DeferredLog(); @Override public void postProcessEnvironment( ConfigurableEnvironment env, SpringApplication app) { log.error("This should be printed"); } @Override public void onApplicationEvent(ApplicationEvent event) { log.replayTo(MyEnvironmentPostProcessor.class); } }
1、在 src/main/resources
下新建META-INF/spring.factories
文件
2、配置
org.springframework.boot.env.EnvironmentPostProcessor=\ com.huan.springcloud.extensionpoint.environmentpostprocessor.CustomEnvironmentPostProcessor
@Component public class PrintCustomizeEnvironmentProperty implements ApplicationRunner { private static final Logger log = LoggerFactory.getLogger(PrintCustomizeEnvironmentProperty.class); @Value("${username}") private String userName; @Override public void run(ApplicationArguments args) { log.info("获取到的 username 的属性值为: {}", userName); } }
参考上方的 3、编写自定义属性并加入Spring Environment中
Anwendungskontexts
aufgerufen werden, und EnvironmentPostProcessor
kann diese Funktion genau implementieren. 🎜username
🎜org.springframework.boot.context.logging.LoggingApplicationListener🎜Was ist also derzeit der Wert von
username
? Hier leihe ich mir ein Bild von Apollo
aus, um dieses Problem zu erklären. 🎜🎜Referenzlink: https://www.apolloconfig.com/#/zh/design/apollo-design🎜🎜🎜🎜Spring hat seit Version 3.1 ConfigurableEnvironment
und PropertySource
hinzugefügt: 🎜rrreee🎜🎜🎜Spring ApplicationContext wird eine Umgebung enthalten (die die Schnittstelle „ConfigurableEnvironment“ implementiert)🎜PropertySource
🎜🎜🎜🎜Property Sources🎜 🎜🎜kann als viele Schlüsselwert-Eigenschaftskonfigurationen verstanden werden🎜🎜🎜Wie aus dem obigen schematischen Diagramm ersichtlich ist, erscheint key
in der ersten PropertySource
Die Priorität ist höher . Im obigen Beispiel ist der Wert von username
in SpringBoot
huan.fu
. 🎜Erhalten Sie die Priorität der Konfigurationseigenschaften
können wir sehen, dass die PropertySource
höher ist Je höher die Ausführung, desto früher muss unsere Konfiguration wirksam werden. 🎜🎜🎜🎜Wie Sie aus dem sehen können Bild oben, SpringBoot Das Laden verschiedener Konfigurationen wird durch EnvironmentPostProcessor
erreicht, und die spezifische Implementierung wird durch ConfigDataEnvironmentPostProcessor
erreicht. Dann schreiben wir selbst eine Implementierungsklasse von EnvironmentPostProcessor
, führen sie dann nach ConfigDataEnvironmentPostProcessor
aus und fügen sie an der ersten Position in Environment
hinzu. 🎜🎜🎜🎜Vier. Implementierung🎜slf4j
zur Ausgabe des Protokolls verwendet wird. Das Protokoll kann derzeit nicht ausgegeben werden, da das Protokollsystem nicht initialisiert wurde. Die Lösung lautet wie folgt:
🎜rrreeesrc/main/resources >META-INF/spring.factories
File🎜🎜🎜🎜2. Konfiguration🎜rrreee3. Schreiben Sie benutzerdefinierte Attribute und fügen Sie sie zur Spring-Umgebung hinzu
. 🎜🎜2. Die Konfiguration wird nicht wirksam, um festzustellen, ob der von @Order oder Ordered zurückgegebene Prioritätswert falsch ist. 🎜看看别的地方是否实现了 EnvironmentPostProcessor或ApplicationContextInitializer或BeanFactoryPostProcessor或BeanDefinitionRegistryPostProcessor等这些接口,在这个里面修改了 PropertySource的顺序。
理解 Spring 获取获取属性的顺序 参考 2、获取配置属性的优先级
如下代码初始化日志系统
org.springframework.boot.context.logging.LoggingApplicationListener
Das obige ist der detaillierte Inhalt vonSo verwenden Sie den EnvironmentPostProcessor von SpringBoot. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!