SPI应用场景及详解

Wesley13
• 阅读 708

java中spi(service provider interface)是jdk内置的一种服务发现机制,可以基于配置,在运行时加载指定服务。java中提供了很多服务提供接口,如jdbc、jndi等。

面对分布式的开发,很多系统之间的调用都是使用rpc直接调用,但是有的时候上游的系统需要调用下游系统很多的接口,导致开发工作量很大。因此上游系统使用spi的方式在jar包中打一个spi接口,让下游的业务实现这个spi接口,上游的系统则可以通用的调用这一个接口就可以使用不同的功能。

下游业务方的实现:

一般我们需要设计如何实现spi接口,那实现的最简单的方式就是implements这个spi接口,这种实现方式不利用后续的扩展。

新建maven  工程

SPI应用场景及详解

服务调用者先定制接口结构,然后提供给服务提供者进行实现,服务提供者按照接口规范,实现,然后在resources/META-INF/services目录,然后在下面新建文件,文件名为接口的路径,图为org.springboot.produce.service.IMsgService

SPI应用场景及详解

文件内容为

SPI应用场景及详解

里面是接口实现类路径

接口:

public interface IMsgService{
   public boolean accept(String msg);
   
   public void sendMsg(String msg);
   
   public String getName();

}

实现KafkaMsgServiceImpl

public class KafkaMsgServiceImpl implements IMsgService{
    @Override
    public boolean accept(java.lang.String msg) {
        // TODO Auto-generated method stub
        if(msg.equals(getName()))
        return true;
        return false;
    }

    @Override
    public void sendMsg(java.lang.String msg) {
        // TODO Auto-generated method stub
        System.out.println("KafkaMsgServiceImpl "+msg);
    }

    @Override
    public java.lang.String getName() {
        // TODO Auto-generated method stub
        return "KafkaMsgServiceImpl";
    }
}

实现QueueMsgServiceImpl

public class QueueMsgServiceImpl implements IMsgService{
    @Override
    public boolean accept(String msg) {
        // TODO Auto-generated method stub
        if(msg.equals(getName()))
        return true;
        return false;
    }

    @Override
    public void sendMsg(String msg) {
        // TODO Auto-generated method stub
        System.out.println("QueueMsgServiceImpl "+msg);
    }

    @Override
    public String getName() {
        // TODO Auto-generated method stub
        return "QueueMsgServiceImpl";
    }

}

然后将springboot-produce工程 maven打包 安装到本地maven仓库  , mvn  install命令,打包后 ,别的工程(bundle)就可以依赖进去,进行调用

SPI应用场景及详解

新建消费者maven工程  springboot-consume

SPI应用场景及详解

pom.xml (springboot-produce为上面服务提供者的jar,这里要依赖引入)

   
    seckill
    springboot-produce
  1.0.0-SNAPSHOT
   

在springboot-consume新建测试类  App

public class App 
{
    public static void main( String[] args )
    {
        ServiceLoader peoples = ServiceLoader.load(IMsgService.class);
        
        Iterator iterator = peoples.iterator();
        while (iterator.hasNext()) {
           IMsgService msgService = iterator.next();
           if(msgService.accept(msgService.getName())){
               msgService.sendMsg("测试");
           }
        }
    }
}
springboot-consume启动类会扫描本地及依赖包下resources/META-INF/*下 有无以该接口org.springboot.produce.service.IMsgService命名的文件。

运行

SPI应用场景及详解

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Easter79 Easter79
2年前
SpringBoot自定义序列化的使用方式
场景及需求:项目接入了SpringBoot开发,现在需求是服务端接口返回的字段如果为空,那么自动转为空字符串。例如:\    {        "id":1,        "name":null    },    {        "id":2,        "name":"x
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
2年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
2年前
SpringBoot自定义序列化的使用方式
场景及需求:项目接入了SpringBoot开发,现在需求是服务端接口返回的字段如果为空,那么自动转为空字符串。例如:\    {        "id":1,        "name":null    },    {        "id":2,        "name":"x
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这