江干区住房和城市建设局网站,如何对网站的文件和资源进行优化?,专业做二手房的网站,合肥网站建设推广服务✅什么是SPI#xff0c;和API有什么区别 ✅典型解析#x1f7e2;拓展知识仓#x1f7e2;如何定义一个SPI#x1f7e2;SPI的实现原理 ✅SPI的应用场景SpringDubbo ✅典型解析 Java 中区分 API和 SPI#xff0c;通俗的进: API和 SPI 都是相对的概念#xff0c;他们的差别只… ✅什么是SPI和API有什么区别 ✅典型解析拓展知识仓如何定义一个SPISPI的实现原理 ✅SPI的应用场景SpringDubbo ✅典型解析 Java 中区分 API和 SPI通俗的进: API和 SPI 都是相对的概念他们的差别只在语义上API 直接被应用开发人员使用SPI 被框架扩展人员使用。 API Application Programming Interface API是一组定义了软件组件之间交与的规则和约定的接口。提供方来制定接口并完成对接口的不同实现调用方只需要调用即可。 SPI Service Provider Interface SPI是一种扩展机制通常用于在应用程序中提供可插拔的实现。调用方可选择使用提供方提供的内置实现也可以自己实现。 请记住这句话: API用于定义调用接口而SPI用于定义和提供可插拔的实现方式 拓展知识仓 如何定义一个SPI 步骤1、定义一组接口(假设是org.foo.demo.IShout)并写出接口的一个或多个实现(假设是org.foo.demo.animal.Dog、org.foo.demo.animal.Cat)。 public interface IShout {void shout();
}public class Cat implements IShout {Overridepublic void shout() {System.out.println(miao miao);}
}public class Dog implements IShout {Overridepublic void shout() {System.out.println(wang wang);}
}步骤2、在src/main/resources/下建立/META-INF/services 目录新增一个以接口命名的文件org.foo.demo.Shout文件)内容是要应用的实现类(这里是org.foo.demo.animal.Dog和org.foo.demo.animal.Cat每行一个类)。 org.foo.demo.animal.Dog org.foo.demo.animal.Cat 步聚3、使用 ServiceLoader 来加载配置文件中指定的实现。 public class SPIMain {public static void main(String[] args) {ServiceLoaderIShout shouts ServiceLoader.load(IShout.class);for (IShout s : shouts) {s .shout( );}}
}结果输出 SPI的实现原理 看ServiceLoader类的签名类的成员变量
public final class ServiceLoaderS implements IterableS {private static final String PREFIX META-INF/services/;// 代表被加载的类或者接口private final ClassS service;//用于定位加载和实例化providers的类加载器private final ClassLoader loader:// 创建ServiceLoader时采用的访问控制上下文private final AccessControlContext acc;//缓存providers按实例化的顺序排列private LinkedHashMapString,S providers new LinkedHashMap();// 懒查找选代器private LazyIterator lookupIterator;....................
}参考具体源码梳理了一下实现的流程如下:
1、应用程序调用ServiceLoader.load方法ServiceLoader.load方法内先创建一个新的ServiceLoader并实例化该类中的成员变量包括: a. loader(ClassLoader类型类加载器) b. acc(AccessControlContext类型访问控制器) c. providers (LinkedHashMap类型用于缓存加载成功的类) d. lookuplterator(实现选代器功能) 2、应用程序通过迭代器接口获取对象实例 A. ServiceLoader先判断成员变量providers对象中(LinkedHashMap类型是否有缓存实例对象如果有缓存直接返回。 B. 如果没有缓存执行类的装载: i、读取META-INF/services/下的配置文件获得所有能被实例化的类的名称 ii、通过反射方法Class.forName0加载类对象并用instance0方法将类实例化 iii、把实例化后的类缓存到providers对象中(LinkedHashMap类型) iv、然后返回实例对象。 ✅SPI的应用场景 概括地说适用于: 调用者根据实际使用需要启用、扩展、或者替换框架的实现策略。比较常见的例子: 1. 数据库驱动加载接口实现类的加载 2. JDBC加载不同类型数据库的驱动 3. 日志门面接口实现类加载 4. SLF4J加载不同提供商的日志实现类 Spring Spring中大量使用了SP1,比如: 对servlet3.0规范对ServletContainerlnitializer的实现、自动类型转换TypeConversion SPI(Converter SPl、Formatter SPI)等 Dubbo Dubbo中也大量使用SPI的方式实现框架的扩展,不过它对Java提供的原生SPI做了封装允许用户扩展实现Filter接口