[SpringBoot]接口的多实现:选择性注入SpringBoot接口的实现类

最近在项目中遇到两种情况,准备写个博客记录一下。

情况说明:Service层一个接口是否可以存在多个具体实现,此时应该如何调用Service(的具体实现)?

其实之前的项目中也遇到过这种情况,只不过我采用的方式是新建了一个Service接口,然后拆分两个实现类,让两个实现类分别实现两个不同的接口,然后在Controller中分别注入两个Service。最近项目中又遇到了这种情况,于是简单研究了一下,一个Service确实是可以有多个实现的,以下介绍4种区分实现类的方式。

以下代码本人均已测试,并对一些情况做了补充(欢迎继续补充)。

文章目录

  • 实现
    • 方式1: 直接使用实现类类名来区分
    • 方式2:`@Qualifier`注解
    • 方式3:设置主实现类
    • 方式4:使用`@Resource`注解
  • 总结
  • 致谢

实现


方式1: 直接使用实现类类名来区分

我在以下代码中直接创建一个接口AaaService,然后让BbbServiceImpl和CccServiceImpl分别来实现AaaService,且BbbServiceImpl和CccServiceImpl注册为Service。

public interface AaaService {
    String say();
}
@Service
public class BbbServiceImpl implements AaaService {
    @Override
    public String say() {
        System.out.println("我是BBB");
        return "我是BBB";
    }
}
@Service
public class CccServiceImpl implements AaaService {
    @Override
    public String say() {
        System.out.println("我是CCC");
        return "我是CCC";
    }
}

以上代码AAA就有了两个Service实现类,尽管我没有遵循一些明明规则,但是BBB与CCC确实是AAA的实现类,且它们三个都是Service层的组件,接下来我在Controller中注入这两个组件。

@RestController
@RequestMapping("/test")
public class TestController {

    @Autowired
    private AaaService bbbServiceImpl;

    @Autowired
    private AaaService cccServiceImpl;

    @GetMapping("/b")
    public String BBB(){
        return bbbServiceImpl.say();
    }

    @GetMapping("/c")
    public String CCC(){
        return cccServiceImpl.say();
    }

}

这时候我们启动项目,通过浏览器分别访问/test/b/test/c地址,情况如下:

  • /test/b:我是BBB(且后端控制台也输出:我是BBB)
  • /test/c:我是CCC(且后端控制台也输出:我是CCC)

因此,我们确实在Controller层同时注入了一个Service的两个实现类。



方式2:@Qualifier注解

通常情况下,如果注入的B与C的变量名没有遵循与实现类同名(小驼峰)的命名方式的话,项目会启动失败,并且报错如下:

Action:

Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed

表现如下:

在这里插入图片描述

这是在提示我们使用@Qualifier注解来区分两个实现类(使用方式一也是可以的)。在一些情况下,我们虽然一个Service有多个实现类,但是我们依旧想用最传统的方式,也就是接口名小驼峰的方式来作为它的实现类来使用,但是很明显,像是下面这段代码在该情况下是不能生效的,因为我们并没有一个叫AaaService的实现类。

@RestController
public class TestController {
  @Autowired
  private AaaService aaaService;
}

上面这段代码运行起来之后就会报我们上面说到的报错,该如何解决呢??只需要在@Autowired上面加上一个注解就可以,写法如下:

@RestController
public class TestController {
  @Qualifier("bbbServiceImpl")
  @Autowired
  private AaaService aaaService;
}

注意,注解中写的内容必须是该Service接口的某个实现类的小驼峰名称。

这样一来,我们就在该Controller中注入了BbbServiceImpl,只不过它的名称是aaaService,在后续使用中,我们调用的也是BbbServiceImpl种的方法。



方式3:设置主实现类

通过以上两种方式,我们已经可以实现自由的选择我们具体要注入的是哪一个实现类,但是如果业务中有这样的一个场景:我默认下只使用A实现类,BC实现类是我在特殊情况下才去使用的。在这种情况下,我希望能有一种“默认”机制的出现,该如何实现呢?请看以下代码:

public interface AaaService  {
	// ...  
}

@Service
@Primary // 注意这一个注解
public interface BbbServiceImpl implements AaaService {
  // Aaa的实现 ...
}

@Service
public interface CccServiceImpl implements AaaService {
  // Aaa的实现 ...
}

在以上代码中,由于B实现上面有一个@Primary注解,因此该实现类会被当做是A接口的主要实现类,这时候我若是未指明具体使用的是哪一个实现类(就不会报错啦),就会默认用的是B实现类。

这时候又出现了一个小问题,如果这时候我想用C,该怎么办呢?换种方式问一下:命名这个东西是随便写的,如果我们恰巧如方式1中所说的,我变量名命名成了cccServiceImpl,它注入的是B还是C呢?答案揭晓:是B在此种情况下,用命名来区分实现类的方式已经失效了,要是想使用其他的实现类,只能使用方式二中的@Qualifier注解来区分。



方式4:使用@Resource注解

正如方式3中所说,我们设置了@Primary之后,已经无法通过@Autowired注解去控制注入哪个实现类了,但是有没有一种方法还是可以用名称去选择呢?当然可以,那就是@Resource注解。

因为Resource注解默认使用名称进行依赖注入,所以情况3中的代码不变,我们对Controller做如下修改:

@RestController
public class TestController {
  
  @Resource
  AaaService cccServiceImpl;
  
  @GetMapping("/test")
  public String test() {
    return cccServiceImpl.say();
  }
  
}

在以上代码中,虽然我们已经对AaaService设置了默认实现类BbbServiceImpl,但是使用@Resource注解注入了名称为cccServiceImpl的实现,这时候访问/test,得到的字符串依旧是"我是CCC"。





总结

  1. 结论:一个Service可以有多个实现,且我们可以手动选择具体使用的哪个实现
  2. 使用@Autowired且没有做过多操作(如设置默认实现)的情况下,我们可以使用注入的变量名称(一定是小驼峰)的方式选择变量名,也可以在@Autowired上面添加@Qualifier("实现类的小驼峰")的方式来选择具体实现类。
  3. 我们可以在接口实现类上写上@Primary注解来认证该实现类为主实现类,在该情况下,如果不特意指定是哪个实现类,则一律认为是默认实现类。
  4. 当指定了默认实现类之后,使用@Autowired+小驼峰名称的注入方式,已经无法指定特定实现类,只能使用``@Qualifier(“实现类的小驼峰”)@+@Autowired@Resource + 小驼峰命名`的方式进行区分。





致谢

感谢 [CSDN | springboot中一个service接口多个实现类,如何注入]

感谢 [稀土掘金 | spring接口多实现类,选择性注入的4种解决方案]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/301750.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

企业一体化管理建设:制造业如何应对信息孤岛和流程断点?-亿发

未来,制造业将成为市场经济的重要支柱。缺乏制造业的支持,整个供应链将受到市场波动的冲击。因此,改革传统的制造业生产和管理方式是市场经济转型的不可或缺的条件。数字化转型趋向于多领域、多行业,企业若要实现长远发展&#xf…

服务器故障与管理口与raid

一,服务器常见故障 1,系统不停重启进入不了系统 排查是否是硬件故障,系统盘是否损坏(硬盘灯红色,黄色,绿色) 查看系统第一启动项是那种方式(硬盘 网络网卡 光驱 U盘) bios 是否双系统&#x…

面向对象软件设计与分析40讲(36)软件开发过程模型之增量模型

文章目录 1 概念2 优点3 缺点4 适用范围1 概念 增量模型强调将整个项目划分为多个增量或阶段,并在每个增量中逐步构建和交付系统的功能。每个增量是对系统的一个部分进行开发、测试和交付,形成一个可用的子系统。 以下是增量过程模型的主要特点和步骤: 划分增量:根据项目…

2024前端炫酷源码分享(附效果图及在线演示)

分享10款非常有趣的前端特效源码 其中包含css动画特效、js原生特效、svg特效以及小游戏等 下面我会给出特效样式图或演示效果图 但你也可以点击在线预览查看源码的最终展示效果及下载源码资源 GSAP-火箭动画特效 GSAP 火箭动画 当氮气充足的情况下 火箭会冲出 并继续飞行 图片…

基于Spring Boot+Vue的课堂管理系统(前后端分离)

该项目完全免费 介绍 基于Spring BootVue的课堂管理系统。前后端分离。包含教师授课管理、学生选退课、聊天室、签到、笔记管理模块等。 技术架构 SpringBoot MyBatis Redis WebSocket VueCLI Axios Element UI 项目特点: 1、后台使用MyBatis连接数据库&…

2024年1月7日15:09:50

2024年1月7日15:09:55复习:我今天学了有价值的东西,那就是在瓦罗兰特拿到了三杀 2024年1月7日15:11:10学习了如何使用vivopad2的键盘 可以稍微用一下 2024年1月7日15:17:58 学习一个编程的题目 2024年1月7日15:31:27不用机械键盘打字效率就是比不用低…

DC电源模块的安全性能评估及认证标准

BOSHIDA DC电源模块的安全性能评估及认证标准 DC电源模块的安全性能评估和认证标准主要涉及以下方面: 1. 安全标准:DC电源模块需要符合国际电工委员会(IEC)和国家标准的相关规定,如IEC 60950-1(信息技术…

(一)看参考手册学stm32基于hal库,点灯时钟配置

(一)看参考手册学stm32基于hal库,点灯时钟配置 这篇文章主要是个人的学习经验,想分享出来供大家提供思路,如果其中有不足之处请批评指正哈。 废话不多说直接开始主题,本人是基于STM32F407VET6芯片&#xf…

静态网页设计——个人图书馆(HTML+CSS+JavaScript)(dw、sublime Text、webstorm、HBuilder X)

前言 声明:该文章只是做技术分享,若侵权请联系我删除。!! 感谢大佬的视频: https://www.bilibili.com/video/BV1VN4y1q7cz/?vd_source5f425e0074a7f92921f53ab87712357b 源码:https://space.bilibili.co…

Java多线程并发篇----第一篇

系列文章目录 文章目录 系列文章目录前言一、JAVA 并发知识库二、Java中实现多线程有几种方法三、继承 Thread 类四、实现 Runnable 接口。前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,看懂了…

第12课 实现桌面与摄像头叠加

在上一节,我们实现了桌面捕获功能,并成功把桌面图像和麦克风声音发送给对方。在实际应用中,有时候会需要把桌面与摄像头图像叠加在一起发送,这节课我们就来看下如何实现这一功能。 1.备份与修改 备份demo11并修改demo11为demo12…

【产品人卫朋】硬件产品经理:从入门到精通

目录 本文目录 1. 前言说明 2. 内容说明 3. 资料包说明 作者简介 本文目录 1. 前言说明 2. 内容说明 3. 资料包说明 1. 前言说明 本篇内容节选自实体书《硬件产品经理:从入门到精通》。 2. 内容说明 鉴于硬件产品的特殊性,不同产品阶段的时间间…

Gin 项目引入热加载

Gin 项目引入热加载 文章目录 Gin 项目引入热加载一、什么是热加载二、Air2.1 介绍2.2 特性特性:2.3 相关文档2.4 安装推荐使用 install.sh使用 go install 2.5 配置环境变量2.6 使用 三、Fresh3.1 介绍3.2 相关文档3.3 安装与使用 四、bee4.1 介绍4.2 相关文档4.3 …

python统计分析——箱线图(plt.boxplot)

参考资料:用python动手学统计学 使用matplotlib.pyplot.boxplot()函数绘制箱线图 import numpy as np import pandas as pd from matplotlib import pyplot as pltdata_set1np.array([2,3,3,4,4,4,4,5,5,6]) data_set2np.array([[2,3,3,4,4,4,4,5,5,6],[5,6,6,7,7…

ssm基于WEB的文学网的设计与实现+vue论文

基于WEB的文学网的设计与实现 摘要 如今,科学技术的力量越来越强大,通过结合较为成熟的计算机技术,促进了学校、医疗、商城等许多行业领域的发展。为了顺应时代的变化,各行业结合互联网、人工智能等技术,纷纷开展了管…

CentOS 7.6下HTTP隧道代理的安全性考虑

在CentOS 7.6上配置HTTP隧道代理时,安全性是一个不可忽视的重要因素。以下是对HTTP隧道代理安全性的一些关键考虑因素: 1. 加密和数据安全 使用强加密算法:确保您使用的是经过广泛认可和强化的加密算法,如AES-256-GCM。数据完整…

STL容器之vector基本操作

目录 vector基本操作 vector构造函数 vector的遍历操作 1.重载[ ]进行遍历。 2.使用迭代器进行遍历。 3.使用范围for循环进行遍历。 4.使用at成员函数进行遍历 。 vector空间增长 1.size:获取当前元素的个数。 2.capacity:获取能存储的元素的个…

小型洗衣机什么牌子好又便宜?实用的小型洗衣机测评

随着近几年大家对于生活健康都有了更高的要求,迷你内衣裤洗衣机逐渐进入了大家的视野,并且在日常生活中所适用的人群也是比较的广泛。很多研究调查表明普通的手洗内衣裤没有办法完全清除细菌,而机洗能够去除绝大部分细菌。但是机洗对于洗衣机…

1.3号io网络

文件IO 1.文件IO是基于系统调用 2.程序每进行一次系统调用,就会从用户空间向内核空间进行一次切换,执行效率较慢 3.目的:由于后期进程间通信,如管道、套接字通信,都使用的是文件IO,所以引入文件IO操作的…

Docker安装Centos8系统

引言:最小安装版Centos8系统安装docker软件安装记录 官网安装教程:https://dockerdocs.cn/engine/install/centos/index.html 操作系统镜像版本 CentOS-Stream-8-x86_64-latest-boot.iso 第一步:更新yum yum -y update第二步:…