Spring同类型多个Bean的注入

简介: Spring 容器中的 Bean 之间会有很多依赖关系,在注入依赖的时候,容器需要明确知道注入的是哪一个 Bean。


一、类型注入冲突



Spring 容器中的 Bean 依赖可以通过名称注入,或者类型注入。

通过名称注入

名称注入会指定一个明确的 Bean 名称,容器不允许注册相同名称的 Bean,所以不会有任何问题。

通过类型注入

通过类型注入的时候,有时会因为多个 Bean 的类型相同而产生冲突。例如:

  • 同一类型注册多个不同名称的 Bean
  • 抽象类型注册多个不同实现类的 Bean

这种情况下,容器不知道该注入哪个会抛出 NoUniqueBeanDefinitionException 异常。


二、解决冲突



假设在项目中定义一个 BeanService 接口,基于该接口有三个实现类 OneServiceImplTwoServiceImplThreeServiceImpl,三个实现类都由 Spring 容器管理。

微信图片_20220519205803.png

在项目中通过 BeanService 接口的类型注入,会产生冲突抛出异常。

// BeanService的三个实现类注册Bean
@Configuration
public class AppConfig {
    @Bean
    public BeanService oneServiceImpl() {
        return new OneServiceImpl();
    }
    @Bean
    public BeanService twoServiceImpl() {
        return new TwoServiceImpl();
    }
    @Bean
    public BeanService threeServiceImpl() {
        return new ThreeServiceImpl();
    }
}
// 通过接口的类型注入会抛出异常
public class ServiceTest {
    @Autowired
    private BeanService beanService;
}


1. 注入主要的


注册 Bean 的时候,使用 @Primary 指定一个 Bean 为主要的,存在冲突时默认选择主要的 Bean。

@Configuration
public class AppConfig {
    @Bean
    @Primary
    public BeanService oneServiceImpl() {
        return new OneServiceImpl();
    }
    // ......
}

@Primary 注解也可和 @Component 等注解一起使用。


2. 注入指定的


注入 Bean 的时候,使用 @Qualifier 指定具体 Bean 的名称,通过名称注入解决冲突。

public class ServiceTest {
    @Autowired
    @Qualifier("oneServiceImpl")
    private BeanService beanService;
    // ......
}

也可以直接通过字段名称来指定具体 Bean 的名称,来解决冲突。

public class ServiceTest {
    @Autowired
    private BeanService oneServiceImpl;
    // ......
}

上面两种方法同样适用于构造器注入和 Setter 方法注入。


三、注入多个 Bean



在实际应用中,如果需要注入符合类型的所有 Bean,可以使用集合类型来注入。

集合类型的注入同样适用于字段注入、构造器注入和 Setter 方法注入。


1. 注入集合


通过数组来注入一种类型的所有 Bean。

public class ServiceTest {
    @Autowired
    private BeanService[] beanServiceArr;
    // ......
}

通过 List 来注入一种类型的所有 Bean。

public class ServiceTest {
    @Autowired
    private List<BeanService> beanServiceList;
    // ......
}

通过 Set 来注入一种类型的所有 Bean。

public class ServiceTest {
    @Autowired
    private Set<BeanService> beanServiceSet;
    // ......
}


2. 注入 Map


通过 Map 来注入一种类型的所有 Bean,Key 的类型固定为 String

Key 存储 Bean 的名称,Value 存储 Bean 的实例。

public class ServiceTest {
    @Autowired
    private Map<String, BeanService> beanServiceMap;
    // ......
}


3. Bean 的顺序


注册 Bean 的时候可以使用 @Order 注解来指定 Bean 的权重(或顺序)。

在使用有序集合(数组或 List)注入的时候,会根据权重来排序。

@Configuration
public class AppConfig {
    @Bean
    @Order(1)
    public BeanService oneServiceImpl() {
        return new OneServiceImpl();
    }
    @Bean
    @Order(3)
    public BeanService twoServiceImpl() {
        return new TwoServiceImpl();
    }
    @Bean
    @Order(2)
    public BeanService threeServiceImpl() {
        return new ThreeServiceImpl();
    }
}

上面配置类注册的 Bean 使用数组或 List 注入时,注入集合类型的元素顺序为:

0 = {OneServiceImpl@1522}
1 = {ThreeServiceImpl@1527}
2 = {TwoServiceImpl@1528}

@Order 注解也可和 @Component 等注解一起使用。


四、附录



1. 常用注解


注解 描述
@Primary 指定主要的 Bean,存在注入冲突时默认注入的 Bean
@Qualifier 指定注入 Bean 的名称
@Order 指定注册同类型的 Bean 的权重(或顺序),值越小,权重越大。


2. 示例代码


Gitee 仓库:

https://gitee.com/code_artist/spring

项目模块:

spring-ioc

示例路径:

cn.codeartist.spring.bean.multi

目录
相关文章
|
2月前
|
Java Spring
在使用Spring的`@Value`注解注入属性值时,有一些特殊字符需要注意
【10月更文挑战第9天】在使用Spring的`@Value`注解注入属性值时,需注意一些特殊字符的正确处理方法,包括空格、引号、反斜杠、新行、制表符、逗号、大括号、$、百分号及其他特殊字符。通过适当包裹或转义,确保这些字符能被正确解析和注入。
108 3
|
2月前
|
Java 测试技术 程序员
为什么Spring不推荐@Autowired用于字段注入?
作为Java程序员,Spring框架在日常开发中使用频繁,其依赖注入机制带来了极大的便利。然而,尽管@Autowired注解简化了依赖注入,Spring官方却不推荐在字段上使用它。本文将探讨字段注入的现状及其存在的问题,如难以进行单元测试、违反单一职责原则及易引发NPE等,并介绍为何Spring推荐构造器注入,包括增强代码可读性和维护性、方便单元测试以及避免NPE等问题。通过示例代码展示如何将字段注入重构为构造器注入,提高代码质量。
104 1
|
1月前
|
缓存 Java Spring
实战指南:四种调整 Spring Bean 初始化顺序的方案
本文探讨了如何调整 Spring Boot 中 Bean 的初始化顺序,以满足业务需求。文章通过四种方案进行了详细分析: 1. **方案一 (@Order)**:通过 `@Order` 注解设置 Bean 的初始化顺序,但发现 `@PostConstruct` 会影响顺序。 2. **方案二 (SmartInitializingSingleton)**:在所有单例 Bean 初始化后执行额外的初始化工作,但无法精确控制特定 Bean 的顺序。 3. **方案三 (@DependsOn)**:通过 `@DependsOn` 注解指定 Bean 之间的依赖关系,成功实现顺序控制,但耦合性较高。
实战指南:四种调整 Spring Bean 初始化顺序的方案
|
7天前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
15 1
|
1月前
|
Java 开发者 Spring
Spring高手之路24——事务类型及传播行为实战指南
本篇文章深入探讨了Spring中的事务管理,特别是事务传播行为(如REQUIRES_NEW和NESTED)的应用与区别。通过详实的示例和优化的时序图,全面解析如何在实际项目中使用这些高级事务控制技巧,以提升开发者的Spring事务管理能力。
46 1
Spring高手之路24——事务类型及传播行为实战指南
|
2月前
|
XML Java 数据格式
Spring从入门到入土(bean的一些子标签及注解的使用)
本文详细介绍了Spring框架中Bean的创建和使用,包括使用XML配置文件中的标签和注解来创建和管理Bean,以及如何通过构造器、Setter方法和属性注入来配置Bean。
75 9
Spring从入门到入土(bean的一些子标签及注解的使用)
|
2月前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
2月前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细解析Spring Bean的生命周期及其核心概念,并深入源码分析。Spring Bean是Spring框架的核心,由容器管理其生命周期。从实例化到销毁,共经历十个阶段,包括属性赋值、接口回调、初始化及销毁等。通过剖析`BeanFactory`、`ApplicationContext`等关键接口与类,帮助你深入了解Spring Bean的管理机制。希望本文能助你更好地掌握Spring Bean生命周期。
118 1
|
2月前
|
Java Spring
获取spring工厂中bean对象的两种方式
获取spring工厂中bean对象的两种方式
46 1
|
2月前
|
缓存 Java Spring
源码解读:Spring如何解决构造器注入的循环依赖?
本文详细探讨了Spring框架中的循环依赖问题,包括构造器注入和字段注入两种情况,并重点分析了构造器注入循环依赖的解决方案。文章通过具体示例展示了循环依赖的错误信息及常见场景,提出了三种解决方法:重构代码、使用字段依赖注入以及使用`@Lazy`注解。其中,`@Lazy`注解通过延迟初始化和动态代理机制有效解决了循环依赖问题。作者建议优先使用`@Lazy`注解,并提供了详细的源码解析和调试截图,帮助读者深入理解其实现机制。
57 1