休言女子非英物,夜夜龙泉壁上鸣。这篇文章主要讲述Java面试官:如何正确使用 Spring 注入集合类型?相关的知识,希望能为你提供帮助。
集合类型的自动注入是Spring提供的另外一个强大功能。我们在方便的使用依赖注入的特性时,必须要思考对象从哪里注入、怎么创建、为什么是注入这一个对象的。虽然编写框架的目的是让开发人员无需关心太多底层细节,能专心业务逻辑的开发,但是作为开发人员不能真的无脑去使用框架。
务必学会注入集合等高级用法,让自己有所提升!
现在有一需求:存在多个用户Bean,找出来存储到一个List。
1 注入方式
1.1 收集方式多个用户Bean的定义:
文章图片
有了集合类型的自动注入后,即可收集零散的用户Bean:
文章图片
这样即可完成集合类型注入:
文章图片
但当持续增加一些student时,可能就不喜欢用上述的注入集合类型了,而是这样:
1.2 直接装配方式
文章图片
文章图片
分开玩,大家应该不会有啥问题,若两种方式共存了,会咋样?
运行程序后发现直接装配方式的未生效:
文章图片
这是为啥呢?
2 源码解析就得精通这两种注入风格在Spring分别如何实现的。
2.1 收集装配
DefaultListableBeanFactory#resolveMultipleBeans
private Object resolveMultipleBeans(DependencyDescriptor descriptor, @Nullable String beanName,
@Nullable Set<
String>
autowiredBeanNames, @Nullable TypeConverter typeConverter) {
final Class<
?>
type = descriptor.getDependencyType();
if (descriptor instanceof StreamDependencyDescriptor) {
// 装配stream
return stream;
}
else if (type.isArray()) {
// 装配数组
return result;
}
else if (Collection.class.isAssignableFrom(type) &
&
type.isInterface()) {
// 装配集合
// 获取集合的元素类型
Class<
?>
elementType = descriptor.getResolvableType().asCollection().resolveGeneric();
if (elementType == null) {
return null;
}
// 根据元素类型查找所有的bean
Map<
String, Object>
matchingBeans = findAutowireCandidates(beanName, elementType,
new MultiElementDescriptor(descriptor));
if (matchingBeans.isEmpty()) {
return null;
}
if (autowiredBeanNames != null) {
autowiredBeanNames.addAll(matchingBeans.keySet());
}
// 转化查到的所有bean放置到集合并返回
TypeConverter converter = (typeConverter != null ? typeConverter : getTypeConverter());
Object result = converter.convertIfNecessary(matchingBeans.values(), type);
// ...
return result;
}
else if (Map.class == type) {
// 解析map
return matchingBeans;
}
else {
return null;
}
}
(1)获取集合类型的elementType
目标类型定义为List students,所以元素类型为User:
文章图片
(2)根据元素类型找出所有Bean
有了elementType,即可据其找出所有Bean:
文章图片
(3)将匹配的所有的Bean按目标类型转化
上一步获取的所有的Bean都以java.util.LinkedHashMap.LinkedValues存储,和目标类型大不相同,所以最后按需转化。
本案例中,需转化为List:
文章图片
2.2 直接装配方式DefaultListableBeanFactory#findAutowireCandidates
【Java面试官(如何正确使用 Spring 注入集合类型())】不再赘述。
最后就是根据目标类型直接寻找匹配Bean名称为students的List< Student> 装配给StudentController#students属性。
当同时满足这两种装配方式时,Spring会如何处理呢?
DefaultListableBeanFactory#doResolveDependency
文章图片
显然这两种装配集合的方式不能同存,结合本案例:
- 当使用收集装配时,能找到任一对应Bean,则返回
- 若一个都没找到,才采用直接装配
3 修正务必避免两种方式共存去装配集合!只选用一种方式即可。
比如只使用直接装配:
文章图片
只使用收集方式:
文章图片
如何做到让用户2优先输出呢?
控制spring bean加载顺序:
Bean上使用@Order注解,如@Order(2)。数值越小表示优先级越高。默认优先级最低。
@DependsOn 使用它,可使得依赖的Bean如果未被初始化会被优先初始化。
添加@Order(number)注解,number越小优先级越高,越靠前
声明Student这些Bean时将id=2的Student提到id=1之前
推荐阅读
- 整理全网可视化大屏有关资料,学习大屏设计,大屏原型参考,大屏设计参考
- ECIC演讲精华|如何构建云原生应用下的高性能持久化存储()
- Vue3 + TypeScript + Gin 实现后台权限管理平台
- 用Python生成二维码
- 云原生架构下的持续交付实践
- linux系列之:告诉他,他根本不懂kill
- 绝了!这是我见过最详细的HashMap源码解析!
- JS语言在HarmonyOS应用开发框架中的作用
- Meterpreter入门与精通(十三)(十四)