Spring如何正确注入集合类型
目录
- 1 注入方式
- 1.1 收集方式
- 1.2 直接装配方式
- 2 源码解析
- 2.1 收集装配
- 1 获取集合类型的elementType
- 2 根据元素类型找出所有Bean
- 3 将匹配的所有的Bean按目标类型转化
- 2.2 直接装配方式
- 3 修正
务必学会注入集合等高级用法,让自己有所提升!
现在有一需求:存在多个用户Bean,找出来存储到一个List。
【Spring如何正确注入集合类型】
1 注入方式
1.1 收集方式
多个用户Bean定义:
文章图片
有了集合类型的自动注入后,即可收集零散的用户Bean:
文章图片
这样即可完成集合类型注入:
文章图片
但当持续增加一些user时,可能就不喜欢用上述的注入集合类型了,而是这样:
1.2 直接装配方式
文章图片
文章图片
分开玩,大家应该不会有啥问题,若两种方式共存了,会咋样?
运行程序后发现直接装配方式的未生效:
文章图片
这是为啥呢?
2 源码解析 就得精通这两种注入风格在Spring分别如何实现的。
2.1 收集装配
DefaultListableBeanFactory#resolveMultipleBeans
private Object resolveMultipleBeans(DependencyDescriptor descriptor, @Nullable String beanName,@Nullable Set autowiredBeanNames, @Nullable TypeConverter typeConverter) {final Class> type = descriptor.getDependencyType(); if (descriptor instanceof StreamDependencyDescriptor) {// 装配streamreturn 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; }// 根据元素类型查找所有的beanMap 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) {// 解析mapreturn matchingBeans; }else {return null; }}
1 获取集合类型的elementType 目标类型定义为List users,所以元素类型为User:
文章图片
2 根据元素类型找出所有Bean 有了elementType,即可据其找出所有Bean:
文章图片
3 将匹配的所有的Bean按目标类型转化 上一步获取的所有的Bean都以
java.util.LinkedHashMap.LinkedValues
存储,和目标类型大不相同,所以最后按需转化。本案例中,需转化为List:
文章图片
2.2 直接装配方式
DefaultListableBeanFactory#findAutowireCandidates
不再赘述。
最后就是根据目标类型直接寻找匹配Bean名称为users的
List
装配给userController#users属性。当同时满足这两种装配方式时,Spring会如何处理呢?
DefaultListableBeanFactory#doResolveDependency
文章图片
显然这两种装配集合的方式不能同存,结合本案例:
- 当使用收集装配时,能找到任一对应Bean,则返回
- 若一个都没找到,才采用直接装配
3 修正 务必避免两种方式共存去装配集合!只选用一种方式即可。
比如只使用直接装配:
文章图片
只使用收集方式:
文章图片
如何做到让用户2优先输出呢?
控制spring bean加载顺序:
- Bean上使用@Order注解,如@Order(2)。数值越小表示优先级越高。默认优先级最低。
- @DependsOn 使用它,可使得依赖的Bean如果未被初始化会被优先初始化。
- 添加@Order(number)注解,number越小优先级越高,越靠前声
- 明user这些Bean时将id=2的user提到id=1之前
推荐阅读
- 考研英语阅读终极解决方案——阅读理解如何巧拿高分
- 如何寻找情感问答App的分析切入点
- Activiti(一)SpringBoot2集成Activiti6
- mybatisplus如何在xml的连表查询中使用queryWrapper
- MybatisPlus使用queryWrapper如何实现复杂查询
- SpringBoot调用公共模块的自定义注解失效的解决
- 解决SpringBoot引用别的模块无法注入的问题
- 如何在Mac中的文件选择框中打开系统隐藏文件夹
- 漫画初学者如何学习漫画背景的透视画法(这篇教程请收藏好了!)
- java中如何实现重建二叉树