Heim  >  Artikel  >  Java  >  So verwenden Sie den EnvironmentPostProcessor von SpringBoot

So verwenden Sie den EnvironmentPostProcessor von SpringBoot

PHPz
PHPznach vorne
2023-05-22 09:25:051503Durchsuche

    1. Hintergrund

    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配置中心后,配置中心的属性就可以在程序中使用了,那么这个是怎么实现的呢?配置中心的属性又是何时加载到程序中的呢?那么我们如果找到了这个是怎么实现的是否就可以 从任何地方加载配置属性配置属性的加解密功能呢

    二、需求

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    从上图中得知,我们的需求很简单,即我们自己定义的属性需要比配置文件中的优先级更高。

    三、分析

    1、什么时候向SpringBoot中加入我们自己的配置属性

    当我们想在Bean中使用配置属性时,那么我们的配置属性必须在Bean实例化之前就放入到Spring到Environment中。即我们的接口需要在 application context refreshed 之前进行调用,而 EnvironmentPostProcessor 正好可以实现这个功能。

    2、获取配置属性的优先级

    我们知道在 Spring中获取属性是有优先级的。
    比如我们存在如下配置属性 username

    ├─application.properties
    │   >> username=huan
    ├─application-dev.properties
    │   >> username=huan.fu

    那么此时 username 的值是什么呢?此处借用 Apollo的一张图来说解释一下这个问题。

    参考链接:https://www.apolloconfig.com/#/zh/design/apollo-design

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    Spring从3.1版本开始增加了ConfigurableEnvironmentPropertySource

    ConfigurableEnvironment
    • Spring的ApplicationContext会包含一个Environment(实现ConfigurableEnvironment接口)

    • ConfigurableEnvironment自身包含了很多个PropertySource

    PropertySource

    • 属性源

    • 可以理解为很多个Key - Value的属性配置

    由上方的原理图可知,key在最开始出现的PropertySource中的优先级更高,上面的例子在SpringBootusername的值为huan.fu

    3、何时加入我们自己的配置

    由第二步 获取配置属性的优先级 可知,PropertySource 越靠前越先执行,那么要我们配置生效,就必须放在越前面越好。

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    由上图可知,SpringBoot加载各种配置是通过EnvironmentPostProcessor来实现的,而具体的实现是ConfigDataEnvironmentPostProcessor来实现的。那么我们自己编写一个EnvironmentPostProcessor的实现类,然后在ConfigDataEnvironmentPostProcessor后执行,并加入到 Environment中的第一位即可。

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    四、实现

    1、引入SpringBoot依赖

    <?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>

    2、在application.properties中配置属性

    vim application.properties
    username=huan

    3、编写自定义属性并加入Spring Environment中

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    注意:
    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);
    				    }
    				}

    4、通过SPI使自定义的配置生效

    1、在 src/main/resources下新建META-INF/spring.factories文件

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    2、配置

    org.springframework.boot.env.EnvironmentPostProcessor=\
      com.huan.springcloud.extensionpoint.environmentpostprocessor.CustomEnvironmentPostProcessor

    5、编写测试类,输出定义的 username 属性的值

    @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);
        }
    }

    6、运行结果

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot

    五、注意事项

    1、日志无法输出

    参考上方的 3、编写自定义属性并加入Spring Environment中

    2. Anforderungen

    So verwenden Sie den EnvironmentPostProcessor von SpringBoot
    • Aus der obigen Abbildung wissen wir, dass unsere Anforderung sehr einfach ist, das heißt, die von uns definierten Eigenschaften müssen eine höhere Priorität haben als die in der Konfigurationsdatei.

      3. Analyse🎜

      1. Wann wir unsere eigenen Konfigurationseigenschaften zu SpringBoot hinzufügen

      🎜Wenn wir Konfigurationseigenschaften in einem Bean verwenden möchten, müssen sich unsere Konfigurationseigenschaften im Bean Put befinden es vor der Instanziierung in Spring und Environment einbinden. Das heißt, unsere Schnittstelle muss vor der Aktualisierung des Anwendungskontexts aufgerufen werden, und EnvironmentPostProcessor kann diese Funktion genau implementieren. 🎜

      2. Die Priorität beim Erhalten von Konfigurationseigenschaften

      🎜Wir wissen, dass es im Frühjahr eine Priorität beim Erhalten von Eigenschaften gibt.
      Zum Beispiel haben wir das folgende Konfigurationsattribut 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🎜🎜So verwenden Sie den EnvironmentPostProcessor von SpringBoot🎜🎜Spring hat seit Version 3.1 ConfigurableEnvironment und PropertySource hinzugefügt: 🎜rrreee🎜🎜🎜Spring ApplicationContext wird eine Umgebung enthalten (die die Schnittstelle „ConfigurableEnvironment“ implementiert)🎜
    • 🎜🎜ConfigurableEnvironment selbst enthält viele PropertySources🎜🎜🎜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. 🎜

      3. Wann wir unsere eigene Konfiguration hinzufügen

      🎜Ab dem zweiten Schritt 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. 🎜🎜So verwenden Sie den EnvironmentPostProcessor von SpringBoot🎜🎜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. 🎜🎜So verwenden Sie den EnvironmentPostProcessor von SpringBoot🎜🎜Vier. Implementierung🎜

      1. Führen Sie SpringBoot-Abhängigkeiten ein

      rrreee

      2. Konfigurieren Sie Eigenschaften in application.properties

      rrreeerreee

      3. Schreiben Sie benutzerdefinierte Eigenschaften und fügen Sie sie zur Spring-Umgebung hinzu

      🎜So verwenden Sie den EnvironmentPostProcessor von SpringBoot🎜🎜Hinweis:
      1 gefunden Es gibt keine Ausgabe im Protokoll. Überprüfen Sie, ob 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:🎜rrreee

      4. Sorgen Sie dafür, dass die benutzerdefinierte Konfiguration über SPI wirksam wird.

      🎜1 Erstellen Sie einen neuen src/main/resources >META-INF/spring.factoriesFile🎜🎜 Verwendung des EnvironmentPostProcessor von SpringBoot🎜🎜2. Konfiguration🎜rrreee

      5. Schreiben Sie eine Testklasse und geben Sie den Wert des definierten Benutzernamenattributs aus

      rrreee

      6. Ergebnisse ausführen

      🎜 So verwenden Sie den EnvironmentPostProcessor von SpringBoot🎜🎜5. Hinweise🎜

      1 . Das Protokoll kann nicht ausgegeben werden.

      🎜Beziehen Sie sich auf die Lösung von 3. 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、获取配置属性的优先级

    3、日志系统如何初始化

    如下代码初始化日志系统

    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!

    Stellungnahme:
    Dieser Artikel ist reproduziert unter:yisu.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen