zoukankan      html  css  js  c++  java
  • Java SPI机制

    SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用框架扩展和替换组件。

    系统设计的各个抽象,往往有很多不同的实现方案,在面向的对象的设计里,一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制。
    Java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点类似IOC的思想,就是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要。所以SPI的核心思想就是解耦

     

    Java SPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制。

    典型实例:jdbc的设计

    通常各大厂商(如Mysql、Oracle)会根据一个统一的规范(java.sql.Driver)开发各自的驱动实现逻辑。客户端使用jdbc时不需要去改变代码,直接引入不同的spi接口服务即可。
    Mysql的则是com.mysql.jdbc.Drive,Oracle则是oracle.jdbc.driver.OracleDriver。

    可以参考如下文件配置

    1.代码编写

    既然是spi,那么就必须先定义好接口。其次,就是定义好接口的实现类。

    public interface HelloService {
        void hello();
    }
    
    public class HelloService1 implements HelloService {
        @Override
        public void hello() {
            System.out.println("hello world");
        }
    }
    
    public class HelloService2 implements HelloService {
        @Override
        public void hello() {
            System.out.println("hello chenziyang");
        }
    }

    2.创建一个文件夹

    在项目的srcmain esources下创建META-INFservices目录,还需要将resources目录变更为Resources资源目录

    3.文件夹下增加配置文件

    在上面META-INFservices的目录下再增加一个配置文件,这个文件必须以接口的全限定类名保持一致,例如:com.java.HelloService

    4.配置文件增加描述

    上面介绍spi时说道,除了代码上的接口实现之外,还需要把该实现的描述提供给JDK。那么,此步骤就是在配置文件中撰写接口实现描述。很简单,就是在配置文件中写入具体实现类的全限定类名,如有多个便换行写入。

    com.java.imp.HelloService1
    com.java.imp.HelloService2

    5.使用JDK来载入

    编写main()方法,输出测试接口。使用JDK提供的ServiceLoader.load()来加载配置文件中的描述信息,完成类加载操作。

    public class Test {
        public static void main(String[] args) {
            ServiceLoader<HelloService> loaders = ServiceLoader.load(HelloService.class);
            for (HelloService hello:loaders) {
                hello.hello();
            }
        }
    }

    最后

     

    总结

    优点
    使用Java SPI机制的优势是实现解耦,使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一起。应用程序可以根据实际业务情况启用框架扩展或替换框架组件。

    相比使用提供接口jar包,供第三方服务模块实现接口的方式,SPI的方式使得源框架,不必关心接口的实现类的路径,可以不用通过下面的方式获取接口实现类:

    • 代码硬编码import 导入实现类
    • 指定类全路径反射获取:例如在JDBC4.0之前,JDBC中获取数据库驱动类需要通过Class.forName("com.mysql.jdbc.Driver"),类似语句先动态加载数据库相关的驱动,然后再进行获取连接等的操作
    • 第三方服务模块把接口实现类实例注册到指定地方,源框架从该处访问实例

    通过SPI的方式,第三方服务模块实现接口后,在第三方的项目代码的META-INF/services目录下的配置文件指定实现类的全路径名,源码框架即可找到实现类

  • 相关阅读:
    shell脚本中使用nohup执行命令不生效
    【异常】Could not find artifact com.wm.****:
    【异常】The last packet sent successfully to the server was 0 milliseconds ago. The driver has not received any packets from the server.
    MySQL添加唯一索引
    MacBook Pro实现共享屏幕(多台mac之间)
    【异常】lockfile.AlreadyLocked: ~/airflow/airflow-scheduler.pid is already locked
    CentOS7.2安装Airflow
    Docker
    Docker
    Docker
  • 原文地址:https://www.cnblogs.com/misaki-workshop/p/13727441.html
Copyright © 2011-2022 走看看