이전 프로젝트에서는 Apollo
구성 센터를 사용했는데, Apollo 구성 센터를 도킹한 후 프로그램에서 구성 센터의 속성을 사용할 수 있습니다. 이거 모직물로 구현됐나요? 구성 센터의 속성은 언제 프로그램에 로드되었습니까? 그렇다면 이것이 어떻게 구현되는지 알아낸다면 어디서나 구성 속성을 로드
하고 구성 속성의 기능을 암호화 및 해독
할 수 있을까요? 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中
애플리케이션 컨텍스트가 새로 고쳐지기
전에 인터페이스를 호출해야 하며 EnvironmentPostProcessor
가 이 기능을 정확하게 구현할 수 있습니다. 🎜username
🎜org.springframework.boot.context.logging.LoggingApplicationListener🎜 구성 속성이 있습니다. 그러면 현재
username
의 값은 무엇입니까? 이 문제를 설명하기 위해 Apollo
의 사진을 빌려왔습니다. 🎜🎜참조 링크: https://www.apolloconfig.com/#/zh/design/apollo-design🎜🎜🎜🎜Spring에는 버전 3.1부터 ConfigurableEnvironment
및 PropertySource
가 추가되었습니다. 🎜rrreee🎜🎜🎜Spring ApplicationContext는 환경 포함(ConfigurableEnvironment 인터페이스 구현)🎜PropertySource
🎜🎜🎜🎜Property Sources🎜 🎜🎜can 많은 Key-Value 속성 구성으로 이해됩니다🎜🎜🎜위의 도식에서 볼 수 있듯이 key
는 첫 번째 PropertySource
에 나타납니다. 우선 순위가 더 높습니다. 위의 예에서 SpringBoot
의 username
값은 huan.fu
입니다. 🎜구성 속성의 우선순위 가져오기
에서 PropertySource
가 더 높은 것을 확인할 수 있습니다. 실행이 높을수록 구성이 적용되려면 가능한 한 빨리 배치되어야 합니다. 🎜🎜🎜🎜에서 볼 수 있듯이 위 그림에서 SpringBoot의 다양한 구성 로딩은 EnvironmentPostProcessor
를 통해 이루어지며, 구체적인 구현은 ConfigDataEnvironmentPostProcessor
를 통해 이루어집니다. 그런 다음 EnvironmentPostProcessor
의 구현 클래스를 직접 작성한 다음 ConfigDataEnvironmentPostProcessor
다음에 이를 실행하고 Environment
의 첫 번째 위치에 추가합니다. 🎜🎜🎜🎜4. 구현🎜slf4j
가 사용되었는지 확인하십시오. 이때 로그 시스템이 초기화되지 않았기 때문에 로그를 출력할 수 없습니다. 해결책은 다음과 같습니다:
🎜rrreeesrc/main/resources >META-INF/spring.factories
파일🎜🎜🎜🎜2. Configuration🎜rrreee3. 사용자 정의 속성을 작성하고 이를 Spring 환경에 추가
에서 제공하는 솔루션을 참조하세요. 🎜🎜2. 구성이 적용되지 않습니다. EnvironmentPostProcessor의 우선순위를 확인하여 @Order 또는 Ordered에서 반환된 우선순위 값이 잘못된지 확인하세요. 🎜看看别的地方是否实现了 EnvironmentPostProcessor或ApplicationContextInitializer或BeanFactoryPostProcessor或BeanDefinitionRegistryPostProcessor等这些接口,在这个里面修改了 PropertySource的顺序。
理解 Spring 获取获取属性的顺序 参考 2、获取配置属性的优先级
如下代码初始化日志系统
org.springframework.boot.context.logging.LoggingApplicationListener
위 내용은 SpringBoot의 EnvironmentPostProcessor를 사용하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!