目錄
一、背景
二、需求
三、分析
1、什麼時候加入SpringBoot加入我們自己的設定屬性
2、取得配置屬性的優先權
3、何時加入我們自己的配置
由第二步驟
3、日志系统如何初始化
首頁 Java java教程 SpringBoot的EnvironmentPostProcessor怎麼用

SpringBoot的EnvironmentPostProcessor怎麼用

May 22, 2023 am 09:25 AM
springboot environmentpostprocessor

    一、背景

    之前專案中用到了Apollo配置中心,對接Apollo配置中心後,配置中心的屬性就可以在程式中使用了,那麼這個是怎麼實現的呢?配置中心的屬性又是何時載入到程式中的呢?那我們如果找到了這個是怎麼實現的是否就可以 從任何地方載入配置屬性配置屬性的加解密功能呢

    二、需求

    SpringBoot的EnvironmentPostProcessor怎麼用

    從上圖得知,我們的需求很簡單,也就是我們自己定義的屬性需要比設定檔中的優先順序更高。

    三、分析

    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

    SpringBoot的EnvironmentPostProcessor怎麼用

    Spring從3.1版本開始增加了ConfigurableEnvironmentPropertySource

    ConfigurableEnvironment
    登入後複製
    • Spring的ApplicationContext會包含一個Environment(實作ConfigurableEnvironment介面)

    ConfigurableEnvironment本身包含了許多PropertySource

    PropertySource

    SpringBoot的EnvironmentPostProcessor怎麼用

    屬性來源

    SpringBoot的EnvironmentPostProcessor怎麼用

    SpringBoot的EnvironmentPostProcessor怎麼用


    ##可以理解為很多Key - Value的屬性配置

    由上方的原理圖可知,

    key在最開始出現的 PropertySource中的優先權較高,上面的範例在SpringBoot

    usernameSpringBoot的EnvironmentPostProcessor怎麼用的值為

    huan.fu

    3、何時加入我們自己的配置

    由第二步驟

    取得配置屬性的優先權

    可知,SpringBoot的EnvironmentPostProcessor怎麼用PropertySource

    越靠前越先執行,那麼要我們配置生效,就必須放在越前面越好。

    由上圖可知,SpringBoot載入各種設定是透過EnvironmentPostProcessor來實現的,而具體的實作是

    ConfigDataEnvironmentPostProcessor

    來實現的。然後我們自己寫一個

    EnvironmentPostProcessor
      的實作類,然後在
    • ConfigDataEnvironmentPostProcessor

      後執行,並加入到

      Environment###中的第一位即可。 ###############四、實作######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中################注意:###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###檔案###############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、運行結果###############五、注意事項######1、日誌無法輸出#######參考上方的###3、編寫自訂屬性並加入Spring Environment中###提供的解決方案。 ######2、設定沒有生效檢查############檢查EnvironmentPostProcessor的優先權,看看是否@Order或Ordered傳回的優先權值不對。 ###
    • 看看别的地方是否实现了 EnvironmentPostProcessor或ApplicationContextInitializer或BeanFactoryPostProcessor或BeanDefinitionRegistryPostProcessor等这些接口,在这个里面修改了 PropertySource的顺序。

    • 理解 Spring 获取获取属性的顺序 参考 2、获取配置属性的优先级

    3、日志系统如何初始化

    如下代码初始化日志系统

    org.springframework.boot.context.logging.LoggingApplicationListener
    登入後複製

    以上是SpringBoot的EnvironmentPostProcessor怎麼用的詳細內容。更多資訊請關注PHP中文網其他相關文章!

    本網站聲明
    本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

    熱AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智慧驅動的應用程序,用於創建逼真的裸體照片

    AI Clothes Remover

    AI Clothes Remover

    用於從照片中去除衣服的線上人工智慧工具。

    Undress AI Tool

    Undress AI Tool

    免費脫衣圖片

    Clothoff.io

    Clothoff.io

    AI脫衣器

    Video Face Swap

    Video Face Swap

    使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

    熱工具

    記事本++7.3.1

    記事本++7.3.1

    好用且免費的程式碼編輯器

    SublimeText3漢化版

    SublimeText3漢化版

    中文版,非常好用

    禪工作室 13.0.1

    禪工作室 13.0.1

    強大的PHP整合開發環境

    Dreamweaver CS6

    Dreamweaver CS6

    視覺化網頁開發工具

    SublimeText3 Mac版

    SublimeText3 Mac版

    神級程式碼編輯軟體(SublimeText3)

    熱門話題

    Java教學
    1658
    14
    CakePHP 教程
    1415
    52
    Laravel 教程
    1309
    25
    PHP教程
    1257
    29
    C# 教程
    1231
    24
    Springboot怎麼整合Jasypt實現設定檔加密 Springboot怎麼整合Jasypt實現設定檔加密 Jun 01, 2023 am 08:55 AM

    Jasypt介紹Jasypt是一個java庫,它允許開發員以最少的努力為他/她的專案添加基本的加密功能,並且不需要對加密工作原理有深入的了解用於單向和雙向加密的高安全性、基於標準的加密技術。加密密碼,文本,數字,二進位檔案...適合整合到基於Spring的應用程式中,開放API,用於任何JCE提供者...添加如下依賴:com.github.ulisesbocchiojasypt-spring-boot-starter2. 1.1Jasypt好處保護我們的系統安全,即使程式碼洩露,也可以保證資料來源的

    怎麼在SpringBoot中使用Redis實現分散式鎖 怎麼在SpringBoot中使用Redis實現分散式鎖 Jun 03, 2023 am 08:16 AM

    一、Redis實現分散式鎖原理為什麼需要分散式鎖在聊分散式鎖之前,有必要先解釋一下,為什麼需要分散式鎖。與分散式鎖相對就的是單機鎖,我們在寫多執行緒程式時,避免同時操作一個共享變數產生資料問題,通常會使用一把鎖來互斥以保證共享變數的正確性,其使用範圍是在同一個進程中。如果換做是多個進程,需要同時操作一個共享資源,如何互斥?現在的業務應用通常是微服務架構,這也意味著一個應用會部署多個進程,多個進程如果需要修改MySQL中的同一行記錄,為了避免操作亂序導致髒數據,此時就需要引入分佈式鎖了。想要實現分

    SpringBoot怎麼整合Redisson實現延遲隊列 SpringBoot怎麼整合Redisson實現延遲隊列 May 30, 2023 pm 02:40 PM

    使用場景1、下單成功,30分鐘未支付。支付超時,自動取消訂單2、訂單簽收,簽收後7天未進行評估。訂單超時未評價,系統預設好評3、下單成功,商家5分鐘未接單,訂單取消4、配送超時,推播簡訊提醒…對於延時比較長的場景、即時性不高的場景,我們可以採用任務調度的方式定時輪詢處理。如:xxl-job今天我們採

    springboot讀取檔案打成jar包後存取不到怎麼解決 springboot讀取檔案打成jar包後存取不到怎麼解決 Jun 03, 2023 pm 04:38 PM

    springboot讀取文件,打成jar包後訪問不到最新開發出現一種情況,springboot打成jar包後讀取不到文件,原因是打包之後,文件的虛擬路徑是無效的,只能通過流去讀取。文件在resources下publicvoidtest(){Listnames=newArrayList();InputStreamReaderread=null;try{ClassPathResourceresource=newClassPathResource("name.txt");Input

    Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Jun 02, 2023 am 11:07 AM

    在Springboot+Mybatis-plus不使用SQL語句進行多表添加操作我所遇到的問題準備工作在測試環境下模擬思維分解一下:創建出一個帶有參數的BrandDTO對像模擬對後台傳遞參數我所遇到的問題我們都知道,在我們使用Mybatis-plus中進行多表操作是極其困難的,如果你不使用Mybatis-plus-join這一類的工具,你只能去配置對應的Mapper.xml文件,配置又臭又長的ResultMap,然後再寫對應的sql語句,這種方法雖然看上去很麻煩,但具有很高的靈活性,可以讓我們

    SpringBoot與SpringMVC的比較及差別分析 SpringBoot與SpringMVC的比較及差別分析 Dec 29, 2023 am 11:02 AM

    SpringBoot和SpringMVC都是Java開發中常用的框架,但它們之間有一些明顯的差異。本文將探究這兩個框架的特點和用途,並對它們的差異進行比較。首先,我們來了解一下SpringBoot。 SpringBoot是由Pivotal團隊開發的,它旨在簡化基於Spring框架的應用程式的建立和部署。它提供了一種快速、輕量級的方式來建立獨立的、可執行

    SpringBoot怎麼自訂Redis實作快取序列化 SpringBoot怎麼自訂Redis實作快取序列化 Jun 03, 2023 am 11:32 AM

    1.自訂RedisTemplate1.1、RedisAPI預設序列化機制基於API的Redis快取實作是使用RedisTemplate範本進行資料快取操作的,這裡開啟RedisTemplate類,查看該類別的源碼資訊publicclassRedisTemplateextendsRedisAccessorimplementsRedisOperations,BeanClassLoaderAware{//聲明了value的各種序列化方式,初始值為空@NullableprivateRedisSe

    springboot怎麼取得application.yml裡值 springboot怎麼取得application.yml裡值 Jun 03, 2023 pm 06:43 PM

    在專案中,很多時候需要用到一些配置信息,這些信息在測試環境和生產環境下可能會有不同的配置,後面根據實際業務情況有可能還需要再做修改。我們不能將這些設定在程式碼中寫死,最好是寫到設定檔中,例如可以把這些資訊寫到application.yml檔案中。那麼,怎麼在程式碼裡取得或使用這個位址呢?有2個方法。方法一:我們可以透過@Value註解的${key}即可取得設定檔(application.yml)中和key對應的value值,這個方法適用於微服務比較少的情形方法二:在實際專案中,遇到業務繁瑣,邏

    See all articles