Spring如何正确注入集合类型

目录
  • 1 注入方式
    • 1.1 收集方式
    • 1.2 直接装配方式
  • 2 源码解析
    • 2.1 收集装配
      • 1 获取集合类型的elementType
      • 2 根据元素类型找出所有Bean
      • 3 将匹配的所有的Bean按目标类型转化
    • 2.2 直接装配方式
  • 3 修正

集合类型的自动注入是Spring提供的另外一个强大功能。我们在方便的使用依赖注入的特性时,必须要思考对象从哪里注入、怎么创建、为什么是注入这一个对象的。虽然编写框架的目的是让开发人员无需关心太多底层细节,能专心业务逻辑的开发,但是作为开发人员不能真的无脑去使用框架。
务必学会注入集合等高级用法,让自己有所提升!

现在有一需求:存在多个用户Bean,找出来存储到一个List。

1 注入方式

1.1 收集方式

多个用户Bean定义:

有了集合类型的自动注入后,即可收集零散的用户Bean:

这样即可完成集合类型注入:

但当持续增加一些user时,可能就不喜欢用上述的注入集合类型了,而是这样:

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 users,所以元素类型为User:

2 根据元素类型找出所有Bean

有了elementType,即可据其找出所有Bean:

3 将匹配的所有的Bean按目标类型转化

上一步获取的所有的Bean都以java.util.LinkedHashMap.LinkedValues存储,和目标类型大不相同,所以最后按需转化。

本案例中,需转化为List:

2.2 直接装配方式

DefaultListableBeanFactory#findAutowireCandidates

不再赘述。

最后就是根据目标类型直接寻找匹配Bean名称为users的List<user>装配给userController#users属性。

当同时满足这两种装配方式时,Spring会如何处理呢?

DefaultListableBeanFactory#doResolveDependency

显然这两种装配集合的方式不能同存,结合本案例:

  • 当使用收集装配时,能找到任一对应Bean,则返回
  • 若一个都没找到,才采用直接装配

所以后期以List方式直接添加的user Bean都不生效!

3 修正

务必避免两种方式共存去装配集合!只选用一种方式即可。
比如只使用直接装配:

只使用收集方式:

如何做到让用户2优先输出呢?
控制spring bean加载顺序:

  1. Bean上使用@Order注解,如@Order(2)。数值越小表示优先级越高。默认优先级最低。
  2. @DependsOn 使用它,可使得依赖的Bean如果未被初始化会被优先初始化。
  3. 添加@Order(number)注解,number越小优先级越高,越靠前声
  4. 明user这些Bean时将id=2的user提到id=1之前

到此这篇关于Spring如何正确注入集合类型的文章就介绍到这了,更多相关Spring 集合类型内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Spring实战之注入集合值操作示例

    本文实例讲述了Spring实战之注入集合值操作.分享给大家供大家参考,具体如下: 一 配置 <?xml version="1.0" encoding="GBK"?> <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://www.springframework.org/schema/beans" xsi:schema

  • 详解Spring注入集合(数组、List、Map、Set)类型属性

    注入集合(数组.List.Map.Set)类型属性 (1)创建类,定义数组,list,map,set类型属性,并且生成对应的set方法. (2)在spring配置文件中进行配置. Stu类: package com.Keafmd.spring5.collectiontype; import java.util.Arrays; import java.util.List; import java.util.Map; import java.util.Set; /** * Keafmd * * @C

  • 详解Java的Spring框架中bean的注入集合

    使用value属性和使用<property>标签的ref属性在你的bean配置文件中的对象引用,这两种情况下可以处理单值到一个bean,如果你想通过多元值,如Java Collection类型List, Set, Map 及 Properties.要处理这种情况,Spring提供了四种类型的如下集合的配置元素: 可以使用<list> 或<set> 来连接任何实现java.util.Collection或数组. 会遇到两种情况(a)将收集的直接的值及(b)传递一个bean

  • Spring如何正确注入集合类型

    目录 1 注入方式 1.1 收集方式 1.2 直接装配方式 2 源码解析 2.1 收集装配 1 获取集合类型的elementType 2 根据元素类型找出所有Bean 3 将匹配的所有的Bean按目标类型转化 2.2 直接装配方式 3 修正 集合类型的自动注入是Spring提供的另外一个强大功能.我们在方便的使用依赖注入的特性时,必须要思考对象从哪里注入.怎么创建.为什么是注入这一个对象的.虽然编写框架的目的是让开发人员无需关心太多底层细节,能专心业务逻辑的开发,但是作为开发人员不能真的无脑去使

  • Spring IOC容器Bean管理XML注入集合类型属性

    目录 一.定义数组.list.map.set类型属性 二.配置文件中进行对应配置 三.注入对象集合类型 四.提取注入集合的部分 1. 引入名称空间 util 2. 使用 util 标签完成集合注入的提取 一.定义数组.list.map.set类型属性 创建类.定义数组.list.map.set类型属性,生成对应set方法. package com.pingguo.spring5.collectiontype; import java.util.Arrays; import java.util.L

  • Spring的@Value注入复杂类型(通过@value注入自定义类型)

    之前写了一篇关于Spring的@Value注入的文章<介绍两种SpringBoot读取yml文件中配置数组的方法>. 里面列出了@Value和@ConfigurationProperties的对比,其中有一条是写的@value不支持复杂类型封装(数组.Map.对象等). 但是后来有小伙伴留言说他用@value测试的时候,是可以注入的数组和集合的.于是我就跟着做了一些测试,发现确实可以.但是只有在以,分割的字符串的时候才可以. 为什么用,分割的字符串可以注入数组?于是我就去一步一步的断点去走了一

  • Spring注入Date类型的三种方法总结

    Spring注入Date类型的三种方法总结 测试Bean: public class DateBean { private Date birthday; public Date getBirthday() { return birthday; } public void setBirthday(Date birthday) { this.birthday = birthday; } } 方式1:利用SimpleDateFormat的构造方法注入 <?xml version="1.0&quo

  • 使用spring注入枚举类型作为参数

    目录 spring注入枚举类型作为参数 Spring参数注入 1.通过构造方法实现参数注入 2.通过set方法注入 3.P命名空间注入 spring注入枚举类型作为参数 //定义枚举类型 public enum ReportType { MONTH,WEEK,DAY } //使用枚举类型 public class ReportJob { private ReportType reportType; } //spring配置文件注入 <bean id="DAY" class=&qu

  • Spring依赖注入多种类型数据的示例代码

    目录 Student实体类 StudentsClass实体类 beans.xml 测试 Student实体类 package entity; import java.util.*; /** * @author LeDao * @company * @create 2022-02-13 21:26 */ public class Student { private int id; private String name; private StudentClass studentClass; pri

  • 浅谈spring ioc的注入方式及注入不同的数据类型

    关于Spring-IoC的简单使用参考: spring ioc的简单实例及bean的作用域属性解析 1.通过set方法注入不同数据类型 测试类代码(set方式注入的属性一定要加set方法) /**通过set方法注入示例*/ public class IoC_By_Set { /**注入Integer类型参数*/ private Integer id; /**注入String类型参数*/ private String name; /**注入实体Bean*/ private User user; /

随机推荐