Spring 注入 Bean 到 List / Map 中

简介: 前言 由于在工作中遇到了这样的问题,所以随笔写下来,记录一下。我们平常习惯的注入方式是:@Autowiredprivate 类的名称 对象名称;这种方式是最简单的,也是最实用的。但是有些时候我们就需要将实现了同样接口的Bean注入到Context中,然后可以用 List / Map 接受注入的对象。

前言

由于在工作中遇到了这样的问题,所以随笔写下来,记录一下。
我们平常习惯的注入方式是:
@Autowired
private 类的名称 对象名称;
这种方式是最简单的,也是最实用的。但是有些时候我们就需要将实现了同样接口的Bean注入到Context中,然后可以用 List / Map 接受注入的对象。

正文

想象一下这样的例子,我们需要解析一个字符串,有好几个解析这个字符串的Service,然后在对赢得地方使用对应的Service。如果我们把每一个解析器Service都有一个自己的接口,那么就违背了,他们都在做同一件事,只是做法不同的初衷。如果是有默认解析器,可以使用@Primary注解(这个注解会在实现相同接口的Bean中找到有@Primary,把这个Bean作为优先级最高的注入进来),如果有好多个@Primary, Spring 又会傻傻分不清,到底要注入那个,于是就会报Exception,这个时候我们可以使用@Qualifier具体指定以哪个实体类注入。如果这样的话,当我们在一个地方需要好多个解析器,我们得写好多个@Autowired以及具体指定的@Qualifier。
Spring的厉害之处,就是咱们的烦恼,他都能想得到,于是便有了将注入的Bean 放在List或者Map中这样的 “玩法”。
下面直接上例子:
我创建了一个接口叫做“Parent”然后有三个实现它的类分别为 “ChildA”,“ChildB”,“ChildC”。然后创建了一个Controller类,用于注入那些Bean。目录结构如下:
屏幕快照 2019-01-26 下午10.02.48.png
Parent 类代码如下:

package com.example.injectdemo.model;

public interface Parent {

    public void say();
}

Child类代码如下:

package com.example.injectdemo.model;

import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

@Component("ChildA")
public class ChildA implements Parent{
    @Override
    public void say() {
        System.out.println("I am ChildA");
    }
}

Tips:@Component一定要传递这个bean注入的名称,因为在这个时候spring不会自动帮我们把注入的bean的名称处理好。
Controller类代码如下:

package com.example.injectdemo;

import com.example.injectdemo.model.Parent;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

import java.util.List;
import java.util.Map;
import java.util.Set;

@Controller
@RequestMapping("/test")
public class InjectController {

    @Autowired
    private Map<String, Parent> map;

    @Autowired
    private List<Parent> list;
    @RequestMapping("/say")
    public void sayTest(){
        Set<Map.Entry<String, Parent>> sets =  map.entrySet();
        for(Map.Entry<String,Parent> item:sets){
            System.out.println(item.getKey());
        }

        for(Parent parent : list){
            parent.say();
        }
    }
}

Tips:Map的Key是实现类的名称,Value为具体的类(Value的泛型为接口名称)
启动Application后,访问URL,查看结果,结果为:

ChildA
ChildB
ChildC
I am ChildA
I am childB
I am ChildC

Map我在这里不多说,通过代码,可以看的一清二楚。
List我在多说一点:
如果我们不想让调用的顺序为ChildA->ChildB->ChildC怎么办。
我们可以分别在这三个实现的@Component注解下面加上@Order(number),这里的number 可以填写数字。数字越大,在list中越靠前。
加完之后 ChildA是这样的:

package com.example.injectdemo.model;

import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

@Component("ChildA")
@Order(3)
public class ChildA implements Parent{
    @Override
    public void say() {
        System.out.println("I am ChildA");
    }
}

其他的B,C分别加上 2和1
输出的顺序是:
ChildA
ChildB
ChildC
I am ChildC
I am childB
I am ChildA

目录
相关文章
|
20天前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
20天前
|
XML Java 数据格式
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
Spring 第二节内容补充 关于Bean配置的更多内容和细节 万字详解!
117 18
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
|
7天前
|
XML Java 数据格式
spring复习02,xml配置管理bean
详细讲解了Spring框架中基于XML配置文件管理bean的各种方式,包括获取bean、依赖注入、特殊值处理、属性赋值、集合类型处理、p命名空间、bean作用域及生命周期和自动装配。
spring复习02,xml配置管理bean
|
5天前
|
算法
你对Collection中Set、List、Map理解?
你对Collection中Set、List、Map理解?
16 5
|
7天前
|
XML Java 数据格式
spring复习03,注解配置管理bean
Spring框架中使用注解配置管理bean的方法,包括常用注解的标识组件、扫描组件、基于注解的自动装配以及使用注解后的注意事项,并提供了一个基于注解自动装配的完整示例。
spring复习03,注解配置管理bean
|
2月前
|
缓存 Java 数据库连接
Spring Boot 资源文件属性配置,紧跟技术热点,为你的应用注入灵动活力!
【8月更文挑战第29天】在Spring Boot开发中,资源文件属性配置至关重要,它让开发者能灵活定制应用行为而不改动代码,极大提升了可维护性和扩展性。Spring Boot支持多种配置文件类型,如`application.properties`和`application.yml`,分别位于项目的resources目录下。`.properties`文件采用键值对形式,而`yml`文件则具有更清晰的层次结构,适合复杂配置。此外,Spring Boot还支持占位符引用和其他外部来源的属性值,便于不同环境下覆盖默认配置。通过合理配置,应用能快速适应各种环境与需求变化。
35 0
|
2月前
|
安全 Java 开发者
开发者必看!@Resource与private final的较量,Spring Boot注入技巧大揭秘,你不可不知的细节!
【8月更文挑战第29天】Spring Boot作为热门Java框架,其依赖注入机制备受关注。本文通过对比@Resource(JSR-250规范)和@Autowired(Spring特有),并结合private final声明的字段注入,详细探讨了两者的区别与应用场景。通过示例代码展示了@Resource按名称注入及@Autowired按类型注入的特点,并分析了它们在注入时机、依赖性、线程安全性和单一职责原则方面的差异,帮助开发者根据具体需求选择最合适的注入策略。
35 0
|
2月前
|
Java Spring
|
2月前
|
存储 Java 索引
|
2月前
|
前端开发 Java 开发者
下一篇
无影云桌面