Spring Bean 依赖注入常见错误

使用 @Value更多是用来装配String,而且支持多种强大的装配方式

Spring Bean 依赖注入常见错误_第1张图片

application.properties配置了这样一个属性:

user=admin

password=pass

然后我们在一个Bean中,分别定义两个属性来引用它们:

Spring Bean 依赖注入常见错误_第2张图片

password返回了配置值,但user却不是配置文件的指定值,而是PC用户名。

答疑


有一个正确的,说明 @Value使用姿势没问题,但user为啥不正确?

这就得精通Spring到底如何根据 @Value查询值。

@Value的核心工作流程

DefaultListableBeanFactory#doResolveDependency

@Nullable

public Object doResolveDependency(DependencyDescriptor descriptor, @Nullable String beanName,

@Nullable Set autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException {

// …

Class type = descriptor.getDependencyType();

// 寻找@Value

Object value = getAutowireCandidateResolver().getSuggestedValue(descriptor);

if (value != null) {

if (value instanceof String) {

// 解析Value值

String strVal = resolveEmbeddedValue((String) value);

BeanDefinition bd = (beanName != null && containsBean(beanName) ?

getMergedBeanDefinition(beanName) : null);

value = evaluateBeanDefinitionString(strVal, bd);

}

// 转化Value解析的结果到装配的类型

TypeConverter converter = (typeConverter != null ? typeConverter : getTypeConverter());

try {

return converter.convertIfNecessary(value, type, descriptor.getTypeDescriptor());

}

catch (UnsupportedOperationException ex) {}

}

// …

}

@Value 的工作大体分为以下三个核心步骤。

1 寻找@Value

判断这个属性字段是否标记为@Value:

QualifierAnnotationAutowireCandidateResolver#findValue
  • valueAnnotationType就是 @Value

Spring Bean 依赖注入常见错误_第3张图片

2 解析@Value的字符串值

若一个字段标记了 @Value,则可拿到对应字符串值,然后根据字符串值解析,最终解析的结果可能是一个字符串or对象,取决于字符串怎么写。

3 将解析结果转化为待装配的对象的类型

当拿到上一步生成的结果后,我们会发现可能和我们要装配的类型不匹配。

比如定义的是UUID,而结果是个字符串,此时就会根据目标类型来寻找转化器执行转化:

Spring Bean 依赖注入常见错误_第4张图片

分析可得问题关键在第二步,执行过程:

![](https://img-blog.csdnimg.cn/728d4ba5c3154e2eb952d32609d5

【一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义】

浏览器打开:qq.cn.hn/FTf 免费领取

3995.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBASmF2YUVkZ2Uu,size_20,color_FFFFFF,t_70,g_se,x_16)

这里是在解析嵌入的值,替换掉占位符。使用PropertySourcesPlaceholderConfigurer根据PropertySources替换。

当使用 ${user} 获取替换值时,最终执行的查找并非只在application.property文件。

可以发现如下“源”都是替换的依据:

Spring Bean 依赖注入常见错误_第5张图片

而具体的查找执行,通过

PropertySourcesPropertyResolver#getProperty

获取执行方式

Spring Bean 依赖注入常见错误_第6张图片

在解析Value字符串有顺序,源都存在CopyOnWriteArrayList,启动时就被按序固定下来了,一个一个“源”顺序查找,在其中一源找到后,就直接返回。

查看systemEnvironment源,发现刚好有个user和自定义的重合,且值不是admin。

你可能感兴趣的:(程序员,面试,后端,java)