首页 > 其他分享 >使用Spring的getBeansOfType实现接口多实现类的动态调用

使用Spring的getBeansOfType实现接口多实现类的动态调用

时间:2023-04-14 11:44:30浏览次数:38  
标签:实现 Spring 接口 getBeansOfType SPI 交通 TrafficCode public

 

背景

org.springframework.beans及org.springframework.context这两个包是Spring IoC容器的基础,
其中重要的类有BeanFactory,BeanFactory是IoC容器的核心接口,其职责包括:实例化、定位、配置应用程序中的
对象及建立这些对象间的依赖关系。

ApplicationContext作为BeanFactory的子类,在Bean管理的功能上得到了很大的增强,也更易于与Spring AOP集成使用。
今天我们要讨论的并不是BeanFactory或者ApplicationContext的实现原理,而是对ApplicationContext的一种实际应用方式。

问题的提出

在实际工作中,我们经常会遇到一个接口及多个实现类的情况,并且在不同的条件下会使用不同的实现类。从使用方式上看,有些类似SPI的用法,
但是由于SPI的使用并不是太方便,那么怎么办呢?我们可以借助ApplicationContext的getBeansOfType来实现我们需要的结果。

首先我们看一下这个方法的签名

<T> Map<String, T> getBeansOfType(Class<T> type) throws BeansException;

从上面的代码上我们可以看出来这个方法能返回一个接口的全部实现类(前提是所有实现类都必须由Spring IoC容器管理)。

接下来看看我们遇到的问题是什么?

"假设从A点到B点有多种交通方式,每种交通方式的费用不同,可以根据乘客的需要进行选择"(好吧,我承认这是个非常蹩脚的需求,
但是可以联想一下类似的需求,比如支付方式、快递公司,在页面提供几个选项,业务代码根据选项的不同选择不同的实现类实例进行调用)。

实现

回到我们的例子,按照这个交通方式的需求,我们的设计如下:有一个交通方式的接口,接口有两个方式,一个查询费用、一个查询该交通方式的类型,同时,我们可以用一个枚举类型类标识交通类型。

我们还需要一个工厂类来根据交通类型标识查找该交通类型的Bean实例,从而使用该实例,获得交通类型的详细信息及该交通类型的操作。

 

代码如下:

接口:

/**
 * 交通方式
 */
public interface TrafficMode {
 
    /**
     * 查询交通方式编码
     * @return 编码
     */
    TrafficCode getCode();
 
    /**
     * 查询交通方式的费用,单位:分
     * @return 费用
     */
    Integer getFee();
 
}

 

枚举:

/**
 * 交通类型枚举
 */
public enum TrafficCode {
 
    TRAIN,
    BUS
 
}

 

接口有两个实现类:

/**
 * 汽车方式
 */
@Component
public class BusMode implements TrafficMode {
 
    @Override
    public TrafficCode getCode() {
        return TrafficCode.BUS;
    }
 
    @Override
    public Integer getFee() {
        return 10000;
    }
 
}

 

工厂类:

/**
 * 交通方式工厂类
 */
@Component
public class TrafficModeFactory implements ApplicationContextAware {
 
    private static Map<TrafficCode, TrafficMode> trafficBeanMap;
 
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        Map<String, TrafficMode> map = applicationContext.getBeansOfType(TrafficMode.class);
        trafficBeanMap = new HashMap<>();
        map.forEach((key, value) -> trafficBeanMap.put(value.getCode(), value));
    }
 
    public static <T extends TrafficMode> T getTrafficMode(TrafficCode code) {
        return (T)trafficBeanMap.get(code);
    }
 
}

 

验证

有了上面的代码之后,我们一起通过单元测试来看一下效果,单元测试代码片段如下:

    @Test
    public void testGetTrafficMode() {
        TrafficMode mode = TrafficModeFactory.getTrafficMode(TrafficCode.BUS);
        Assert.assertEquals(mode.getFee().intValue(), 10000);
 
        mode = TrafficModeFactory.getTrafficMode(TrafficCode.TRAIN);
        Assert.assertEquals(mode.getFee().intValue(), 9000);
    }

 

运行之后的结果呢?必然是通过。

关于SPI

文章到这里,有同学可能会问:这和SPI有什么区别呢?SPI同样也能实现同样的功能啊。首先说明一下,SPI是JDK自带的功能,虽然历史是比较久远的了,但是不代表它不好,而且有些场景非SPI不可(比如JDBC)。

我们明确一下SPI是什么以及它的设计是用来做什么的。SPI的全名为Service Provider Interface(服务提供接口),因为这个是针对厂商或者插件的。比较经典的用法就是JDBC,java提供了标准的JDBC接口,每个数据库厂商提供自己的数据库驱动实现。下面是JDBC驱动的接口:

public interface Driver {
    Connection connect(String var1, Properties var2) throws SQLException;
 
    boolean acceptsURL(String var1) throws SQLException;
 
    DriverPropertyInfo[] getPropertyInfo(String var1, Properties var2) throws SQLException;
 
    int getMajorVersion();
 
    int getMinorVersion();
 
    boolean jdbcCompliant();
 
    Logger getParentLogger() throws SQLFeatureNotSupportedException;
}

 

下面是MySQL的JDBC驱动实现的SPI配置:

关于SPI我们点到为止,这里只是要说明SPI和我们前面例子中使用的AP具有不同的适用场景。

在前面的例子里,我们用的是什么呢?Spring的API,API的全称为Application Programming Interface(应用程序编程接口),在一个应用内部,使用API是非常便捷的方式,也是最直接的方式。

总结一下,就是编程中,我们使用API是最多的。当然我们也会使用SPI(虽然我们不是严格意义上的厂商或者插件),使用SPI也是在特定场景下为了解决问题的一种途径。

关于SPI更多的内容,后续会有文章专门介绍。

【附】关于SPI的约定:当服务的提供者,提供了服务接口的一种实现之后,在jar包的META-INF/services/目录里同时创建一个以服务接口命名的文件。该文件里就是实现该服务接口的具体实现类。而当外部程序装配这个模块的时候,就能通过该jar包META-INF/services/里的配置文件找到具体的实现类名,并装载实例化,完成模块的注入。 基于这样一个约定就能很好的找到服务接口的实现类,而不需要再代码里制定。jdk提供服务实现查找的一个工具类:java.util.ServiceLoader,通过其load方法,传入接口便能获得其实现类。

标签:实现,Spring,接口,getBeansOfType,SPI,交通,TrafficCode,public
From: https://www.cnblogs.com/kelelipeng/p/17317845.html

相关文章

  • 半小时实现Java网络爬虫框架
    最近在做一个搜索相关的项目,需要爬取网络上的一些链接存储到索引库中,虽然有很多开源的强大的爬虫框架,但本着学习的态度,自己写了一个简单的网络爬虫,以便了解其中的原理。今天,就为小伙伴们分享下这个简单的爬虫程序!!首先介绍每个类的功能:DownloadPage.java的功能是下载此超链接的页......
  • Mysql_快速返回机制的实现
    MYSQLJDBC快速查询响应的方法,快速返回机制的实现Oracle的快速返回机制,虽然结果集很多,可是它能很快的显示第一个结果,虽然通过MYSQl的客户端可以做到,但是通过JDBC却不行。今天用了1个多小时,终于搞定此问题,希望对广大Java朋友在处理数据库时有个参考。来由:   通过命令行客户端加......
  • 用python和批处理命令实现Markdown内嵌图片
    img.py代码如下importbase64fromPILimportImage,ImageGrabimg_name="C:\\Users\\Lenovo\\Desktop\\grab_clipboard.png"#获取并保存剪贴板图片im=ImageGrab.grabclipboard()ifisinstance(im,Image.Image):#print("Image:size:%s,mode:%......
  • JSP中树状图的代码实现
    需要使用到jquery插件jquery.treeview.js 1、树状结构<ulid="tree"><s:iteratorvalue="#application.topPrivilegeList"><li>${name}<ul><s:iteratorvalue="children"&g......
  • pytest + yaml 框架 -24.单个用例中参数化功能实现
    前言早期版本参数化功能实现只支持在config中全局的地方写parameters关键字,只实现了基本的功能。v1.2.1版本对parameters参数化功能做了进一步的细分,支持在case用例中针对单个用例的参数化了。parameters参数化用例参数化的实现,我设计了2种实现方式参数化方式1:con......
  • 论产品的需求与实现系列之数据平台
         产品的需求与实现系列:     论产品的需求与实现系列之日志系统     论产品的需求与实现系列之监控系统     论产品的需求与实现系列之数据平台     论产品的需求与实现系列之ci持续集成      搭建这一套是一个组合airp......
  • 27-springcloud-config-3-构建 Spring cloud config 配置中心服务端
    构建一个springcloudconfig配置中心按照如下方式进行:1、创建一个普通的SpringBoot项目2、在pom.xml文件中添加如下依赖:<dependency>    <groupId>org.springframework.cloud</groupId>    <artifactId>spring-cloud-config-server</artifactId></dependency>......
  • 跨网摆渡系统如何实现数据安全交换,从而驱动业务流转?
    在这个数据驱动的时代,一次数据泄露就可能影响到数亿甚至数十亿人。数字化转型进一步推动了数据的移动,而随着攻击者加速利用日常生活中的数据依赖性,数据泄露也正随之扩大。一旦边界的防线被攻破或绕过,攻击者就可以在数据中心内部横向移动,而中心内部基本没有安全控制的手段可以阻止......
  • 26-springcloud-config-3-配置中心仓库
    远程git 本地git还有码云码云上创建仓库,gitclone到本地,是一个工程;new一个目录config-server需要在gitee上设置好配置中心,我们通过idea把gitee上的springcloud项目clone到本地,然后再项目下创建一个文件夹config-center,然后在config-center中创建四个配置文件,如下:ap......
  • 【转】基于.net6+gtksharp实现的Linux下的图形界面串口调试工具
    【开源】基于.net6+gtksharp实现的Linux下的图形界面串口调试工具-狼性法则-博客园(cnblogs.com) 背景22年初从上家互联网公司离职以后,充分认识到互联网行业的风险,公司在没有自身稳定产品的情况下,互联网行业就是一个烧钱的行业,支出远远大于收入来源,上家公司就是如......