Maison >Java >javaDidacticiel >Comment utiliser l'EnvironmentPostProcessor de SpringBoot
Le centre de configuration Apollo
a été utilisé dans le projet précédent. Après avoir ancré le centre de configuration Apollo, les propriétés du centre de configuration peuvent être utilisées dans le programme. Alors comment. est-ce mis en œuvre ? Quand les propriétés du centre de configuration ont-elles été chargées dans le programme ? Donc, si nous découvrons comment cela est implémenté, pouvons-nous charger les propriétés de configuration de n'importe où
et crypter et décrypter les fonctions des propriétés de configuration
? 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中
contexte d'application ne soit actualisé
, et EnvironmentPostProcessor
peut exactement implémenter cette fonction. 🎜username
🎜org.springframework.boot.context.logging.LoggingApplicationListener🎜Alors, quelle est la valeur de
username
à ce moment ? Ici, j'emprunte une image à Apollo
pour expliquer ce problème. 🎜🎜Lien de référence : https://www.apolloconfig.com/#/zh/design/apollo-design🎜🎜🎜🎜Spring a ajouté ConfigurableEnvironment
et PropertySource
depuis la version 3.1 : 🎜rrreee🎜🎜🎜Spring ApplicationContext sera contenir un environnement (implémentant l'interface ConfigurableEnvironment)🎜PropertySource
🎜🎜🎜🎜Property Sources🎜 🎜🎜peut être compris comme de nombreuses configurations de propriétés clé-valeur🎜🎜🎜Comme le montre le diagramme schématique ci-dessus, key
apparaît dans le premier PropertySource
. La priorité est plus élevée . Dans l'exemple ci-dessus, la valeur de username
dans SpringBoot
est huan.fu
. 🎜Obtenir la priorité des propriétés de configuration
nous pouvons voir que plus la PropertySource
est élevée. c'est-à-dire que plus il est Exécution, alors pour que notre configuration prenne effet, elle doit être placée le plus tôt possible. 🎜🎜🎜🎜Comme vous pouvez le voir sur le Dans l'image ci-dessus, SpringBoot Le chargement de diverses configurations est réalisé via EnvironmentPostProcessor
, et l'implémentation spécifique est réalisée par ConfigDataEnvironmentPostProcessor
. Ensuite, nous écrivons nous-mêmes une classe d'implémentation de EnvironmentPostProcessor
, puis l'exécutons après ConfigDataEnvironmentPostProcessor
et l'ajoutons à la première position dans Environment
. 🎜🎜🎜🎜Quatre implémentations🎜slf4j
est utilisé pour sortir le journal. À ce stade, le journal ne peut pas être sorti car le système de journalisation n'a pas été initialisé. La solution est la suivante :
🎜rrreeesrc/main/resources<.> >META-INF/spring.factories</.>
Fichier🎜🎜🎜🎜2. Configuration🎜rrreee3 Écrivez des attributs personnalisés et ajoutez-les à Spring Environment
ci-dessus. 🎜🎜2. La configuration ne prend pas effet. Vérifiez la priorité de EnvironmentPostProcessor pour voir si la valeur de priorité renvoyée par @Order ou Ordered est incorrecte. 🎜看看别的地方是否实现了 EnvironmentPostProcessor或ApplicationContextInitializer或BeanFactoryPostProcessor或BeanDefinitionRegistryPostProcessor等这些接口,在这个里面修改了 PropertySource的顺序。
理解 Spring 获取获取属性的顺序 参考 2、获取配置属性的优先级
如下代码初始化日志系统
org.springframework.boot.context.logging.LoggingApplicationListener
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!