SpringBoot yml配置null坑

  • 场景:yml配置变量值是null 解析得到的是" "
  • tip: 这篇最好结合 MBP中yml属性注入及逻辑删除过程一起看下

  • 下面就来分析下原因:
  1. yml解析过程中进行了默认处理
    分析这部分代码一定要搞清楚Iterator逻辑 因为内部增强for默认使用Iterator遍历的
    最好知悉Iterator及Iterable 两者区别
  2. 设置yml属性值时进行了默认处理(@Value是通过setter操作的 还是是直接反射字段赋值的)-反射字段赋值的

结果是: 1
org.springframework.boot.env.OriginTrackedYamlLoader.OriginTrackingConstructor#getValue:

        private Object getValue(Object value) {
            return (value != null) ? value : "";
        }

null:


null

“null”:


"null"

    1. yml|yaml解析及默认处理的部分流程如下:
  1. org.springframework.boot.SpringApplication#run(java.lang.String...)
  2. org.springframework.boot.SpringApplication#prepareEnvironment
  3. org.springframework.boot.SpringApplicationRunListeners#environmentPrepared
  4. org.springframework.boot.context.event.EventPublishingRunListener#environmentPrepared
  5. org.springframework.context.event.SimpleApplicationEventMulticaster#multicastEvent(org.springframework.context.ApplicationEvent)
  6. org.springframework.context.event.SimpleApplicationEventMulticaster#multicastEvent(org.springframework.context.ApplicationEvent, org.springframework.core.ResolvableType)
  7. org.springframework.context.event.SimpleApplicationEventMulticaster#invokeListener
  8. org.springframework.context.event.SimpleApplicationEventMulticaster#doInvokeListener
  9. org.springframework.boot.context.config.ConfigFileApplicationListener#onApplicationEvent
  10. org.springframework.boot.context.config.ConfigFileApplicationListener#onApplicationEnvironmentPreparedEvent
  11. org.springframework.boot.context.config.ConfigFileApplicationListener#postProcessEnvironment
  12. org.springframework.boot.context.config.ConfigFileApplicationListener#addPropertySources
  13. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#load()
  14. org.springframework.boot.context.config.FilteredPropertySource#apply
  15. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#load()
  16. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#load(org.springframework.boot.context.config.ConfigFileApplicationListener.Profile, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentFilterFactory, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentConsumer)
  17. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#load(java.lang.String, java.lang.String, org.springframework.boot.context.config.ConfigFileApplicationListener.Profile, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentFilterFactory, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentConsumer)
  18. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#loadForFileExtension
  19. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#load(org.springframework.boot.env.PropertySourceLoader, java.lang.String, org.springframework.boot.context.config.ConfigFileApplicationListener.Profile, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentFilter, org.springframework.boot.context.config.ConfigFileApplicationListener.DocumentConsumer)
  20. org.springframework.boot.context.config.ConfigFileApplicationListener.Loader#loadDocuments
  21. org.springframework.boot.env.YamlPropertySourceLoader#load
  22. org.springframework.boot.env.OriginTrackedYamlLoader#load
  23. org.springframework.beans.factory.config.YamlProcessor#process(org.springframework.beans.factory.config.YamlProcessor.MatchCallback)
  24. org.springframework.beans.factory.config.YamlProcessor#process(org.springframework.beans.factory.config.YamlProcessor.MatchCallback, org.yaml.snakeyaml.Yaml, org.springframework.core.io.Resource)
  25. java.util.Iterator#next
  26. org.yaml.snakeyaml.constructor.BaseConstructor#getData
  27. org.yaml.snakeyaml.constructor.BaseConstructor#constructDocument
  28. org.springframework.boot.env.OriginTrackedYamlLoader.OriginTrackingConstructor#constructObject
  29. org.yaml.snakeyaml.constructor.BaseConstructor#constructObject
  30. org.yaml.snakeyaml.constructor.BaseConstructor#constructObjectNoCheck
  31. org.yaml.snakeyaml.constructor.SafeConstructor.ConstructYamlMap#construct
  32. org.yaml.snakeyaml.constructor.BaseConstructor#constructMapping
  33. org.yaml.snakeyaml.constructor.SafeConstructor#constructMapping2ndStep
  34. org.yaml.snakeyaml.constructor.BaseConstructor#constructMapping2ndStep
  35. org.springframework.boot.env.OriginTrackedYamlLoader.OriginTrackingConstructor#constructObject
  36. org.yaml.snakeyaml.constructor.BaseConstructor#constructObject
    ....

在org.springframework.boot.env.OriginTrackedYamlLoader.OriginTrackingConstructor#getValue 打个断点 观察具体的调用栈信息


具体的调用栈信息
    1. @Value注解赋值的过程部分流程如下:


      @Value注入的核心代码

结论:@Value是通过反射字段赋值的 并没有进行null的处理操作
org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.AutowiredFieldElement#inject

@Override
        protected void inject(Object bean, @Nullable String beanName, @Nullable PropertyValues pvs) throws Throwable {
            Field field = (Field) this.member;
            Object value;
            if (this.cached) {
                value = resolvedCachedArgument(beanName, this.cachedFieldValue);
            }
            else {
                DependencyDescriptor desc = new DependencyDescriptor(field, this.required);
                desc.setContainingClass(bean.getClass());
                Set<String> autowiredBeanNames = new LinkedHashSet<>(1);
                Assert.state(beanFactory != null, "No BeanFactory available");
                TypeConverter typeConverter = beanFactory.getTypeConverter();
                try {
                    value = beanFactory.resolveDependency(desc, beanName, autowiredBeanNames, typeConverter);
                }
                catch (BeansException ex) {
                    throw new UnsatisfiedDependencyException(null, beanName, new InjectionPoint(field), ex);
                }
                synchronized (this) {
                    if (!this.cached) {
                        if (value != null || this.required) {
                            this.cachedFieldValue = desc;
                            registerDependentBeans(beanName, autowiredBeanNames);
                            if (autowiredBeanNames.size() == 1) {
                                String autowiredBeanName = autowiredBeanNames.iterator().next();
                                if (beanFactory.containsBean(autowiredBeanName) &&
                                        beanFactory.isTypeMatch(autowiredBeanName, field.getType())) {
                                    this.cachedFieldValue = new ShortcutDependencyDescriptor(
                                            desc, autowiredBeanName, field.getType());
                                }
                            }
                        }
                        else {
                            this.cachedFieldValue = null;
                        }
                        this.cached = true;
                    }
                }
            }
            if (value != null) {
                ReflectionUtils.makeAccessible(field);
                field.set(bean, value);
            }
        }
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,941评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,397评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,345评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,851评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,868评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,688评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,414评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,319评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,775评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,945评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,096评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,789评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,437评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,993评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,107评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,308评论 3 372
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,037评论 2 355

推荐阅读更多精彩内容