Spring 注入 Bean 到 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。目录结构如下:
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
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
大数据技术扫盲,你必须会的这些点
虽说人生没有白走的路,新的一年来到,会的还是原来的知识,人的身价就摆在那里,无论怎么折腾,也不会拿到更好的offer。所以在年轻还有拼劲的时候多学学知识,寻找自身的不足,查漏补缺非常重要。**今天小编给大家带来的是绝对的干货!以下是我自己这些年爬过的那些坑。在大数据开发这一块来说还算是比较全面的吧!废话不多说,直接上干货! 1、Java编程技术 Java编程技术是大数据学习的基础,Java是一种强类型语言,拥有极高的跨平台能力,可以编写桌面应用程序、Web应用程序、分布式系统和嵌入式系统应用程序等,是大数据工程师最喜欢的编程工具,因此,想学好大数据,掌握Java基础是必不可少的。 2、Linux命令 对于大数据开发通常是在Linux环境下进行的,相比Linux操作系统,Windows操作系统是封闭的操作系统,开源的大数据软件很受限制,因此
- 下一篇
SQL 中删除超出时间限制的数据,并返回删除数据信息(Mybatis+postgresql)
前言 前一阵子,接到一个活,主要内容是这样的,数据库中存在一些过期的日志(可能是一天前的数据,或者是一个月前的数据,等等),将这些数据删除掉,并且返回这些数据的信息(就是说我得知道自己到底删了哪些数据呀?)。苦于在网上找了好久,终无果,于是清明小长假,便在自己的mac上,简单的搭了个环境,试了一下,谁让自己是初出茅庐的小菜鸟呢。因为看到网上很多都是关于sql如何操作的,但是并没有结合Mybatis的,所以在这里记录一下,方便自己日后使用,也希望可以帮到那些一直寻找的人。 正文 先给大家看一下我自拟的表结构: -- ---------------------------- -- Table structure for eventLog -- ---------------------------- DROP TABLE IF EXISTS "public"."eventLog"; CREATE TABLE "public"."eventLog" ( "id" int4 NOT NULL DEFAULT nextval('"eventLog_id_seq"'::regclass), "st...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS8编译安装MySQL8.0.19
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2整合Redis,开启缓存,提高访问速度
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Hadoop3单机部署,实现最简伪集群
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果