• 容器环境注入Spring属性不一致却能生效


    前言

    最近使用容器部署应用,Spring应用,会注入一些环境变量,然而这些环境变量的大小写和真实的取值差异很大,而且也不是xxx.xxx,而是xxx_xxx,非常奇怪,代码里也没发现原因。通过分析Spring源码发现,原理就是Spring的特殊处理,以及Spring的设计。

    1.注入系统环境变量

    随意写一个demo

    1. import org.junit.jupiter.api.Test;
    2. import org.springframework.core.env.SystemEnvironmentPropertySource;
    3. import java.util.HashMap;
    4. import java.util.Map;
    5. public class EnvPropertiesTest {
    6. @Test
    7. public void getProperties(){
    8. Map map = new HashMap<>();
    9. map.put("SPRING_DEF_DEMO", "hi, I`m a demo");
    10. map.put("SPRING-DEF-DEMO", "hi, I`m a demo2222");
    11. map.put("spring.def.demo", "hi, I`m a demo2222333");
    12. SystemEnvironmentPropertySource propertySource = new SystemEnvironmentPropertySource("sysEnv", map);
    13. System.out.println(propertySource.getProperty("spring.def.demo"));
    14. }
    15. }

    运行后可以得出

     

     跟我们想的一致,那么现在实现特殊操作,把

    map.put("spring.def.demo", "hi, I`m a demo2222333");

    注释掉

     懵了,为啥获取spring.def.demo,却拿到了SPRING_DEF_DEMO

    另外通过SPRING-DEF-DEMO也可以读取到 SPRING_DEF_DEMO的值。这就是容器注入大写下划线,而通过常规方式读取到的现象,实际上大写加下划线可以认为常量定义。

    2.源码分析原理

    原理实际上可以看到是Spring自己处理了。那么Spring是怎么处理的呢,Spring读取环境变量实际上和系统变量是有区别的。

    org.springframework.core.env.StandardEnvironment

    定义在Spring-core包中

    1. protected void customizePropertySources(MutablePropertySources propertySources) {
    2. propertySources.addLast(
    3. new PropertiesPropertySource(SYSTEM_PROPERTIES_PROPERTY_SOURCE_NAME, getSystemProperties()));
    4. propertySources.addLast(
    5. new SystemEnvironmentPropertySource(SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, getSystemEnvironment()));
    6. }

    上面的是系统变量,下面是环境变量

    System.getenv()
    System.getProperties()

    上面的demo,之所以使用

    SystemEnvironmentPropertySource

    就是这里Spring定义决定的,关键在于getProperty方法

    1. public Object getProperty(String name) {
    2. String actualName = resolvePropertyName(name);
    3. if (logger.isDebugEnabled() && !name.equals(actualName)) {
    4. logger.debug("PropertySource '" + getName() + "' does not contain property '" + name +
    5. "', but found equivalent '" + actualName + "'");
    6. }
    7. return super.getProperty(actualName);
    8. }

    核心在于

    String actualName = resolvePropertyName(name);

    真实名称,这就是Spring的环境变量的设计 

    1. protected final String resolvePropertyName(String name) {
    2. Assert.notNull(name, "Property name must not be null");
    3. // 拿到替换了. -为_的name读取,前提是name不能直接取到值数据
    4. String resolvedName = checkPropertyName(name);
    5. if (resolvedName != null) {
    6. return resolvedName;
    7. }
    8. //大写处理,注入大写之所以可以读取,根源就是这里
    9. String uppercasedName = name.toUpperCase();
    10. if (!name.equals(uppercasedName)) { //严谨处理,毕竟eq就没必要进一步处理了
    11. resolvedName = checkPropertyName(uppercasedName); //大写进一步看是否有大写属性值
    12. if (resolvedName != null) {
    13. return resolvedName;
    14. }
    15. }
    16. return name;
    17. }

    继续

    1. private String checkPropertyName(String name) {
    2. // Check name as-is 如果能直接取到属性,就不转义
    3. if (containsKey(name)) {
    4. return name;
    5. }
    6. // Check name with just dots replaced
    7. // 替换.为_
    8. String noDotName = name.replace('.', '_');
    9. if (!name.equals(noDotName) && containsKey(noDotName)) {
    10. return noDotName;
    11. }
    12. // Check name with just hyphens replaced
    13. // 同理替换中划线
    14. String noHyphenName = name.replace('-', '_');
    15. if (!name.equals(noHyphenName) && containsKey(noHyphenName)) {
    16. return noHyphenName;
    17. }
    18. // Check name with dots and hyphens replaced
    19. // 替换.后替换中划线,即2种同时存在
    20. String noDotNoHyphenName = noDotName.replace('-', '_');
    21. if (!noDotName.equals(noDotNoHyphenName) && containsKey(noDotNoHyphenName)) {
    22. return noDotNoHyphenName;
    23. }
    24. // Give up 牛逼注释
    25. return null;
    26. }
    27. private boolean containsKey(String name) {
    28. return (isSecurityManagerPresent() ? this.source.keySet().contains(name) : this.source.containsKey(name));
    29. }

    至此,就知道了容器或者虚拟机注入环境变量,但是spring读取却可以使用常规xxx.xxx小写读取到的原理,Spring充当了转义器的角色。

     

    总结

    因为框架有定制,笔者一开始并不知道为啥能读取成功,而实际上却神奇的读取到环境变量了,翻遍定制代码都没发现哪里处理的,开始怀疑Spring干的,而Spring读取环境变量的代码就是org.springframework.core.env.SystemEnvironmentPropertySource,刚好就找到处理的代码了,明白原理。

  • 相关阅读:
    高等教育的5个数字营销策略
    探索ClickHouse——使用MaterializedView存储kafka传递的数据
    龙芯3a5000下编译nginx源码
    【填坑指南】PHP8报:Unable to load dynamic library ‘zip.so’ 错误
    [nacos]nacos2.x+nginx集群搭建以及过程中遇到的坑
    DADPS-生物素-炔基_CAS:2241685-22-1试剂反应原理
    PHP和JAVA AES加解密问题
    iNFTnews | 国内NFT发展仅限于数字藏品吗?
    并发编程系列之Lock锁可重入性与公平性
    python 在window对exe、注册表、bat、系统服务操作等实例讲解
  • 原文地址:https://blog.csdn.net/fenglllle/article/details/126942480