Java
程序员在日常工作中经常会听到 SPI
,而且很多框架都使用了 SPI
的技术,那么问题来了,到底什么是 SPI
呢?今天就带大家好好了解一下 SPI。
SPI
全称是 Service Provider Interface
,是一种 JDK
内置的动态加载实现扩展点的机制,通过 SPI
技术我们可以动态获取接口的实现类,不用自己来创建。
这里提到了接口和实现类,那么 SPI
技术上具体有哪些技术细节呢?
SPI
机制,必须有一个与接口同名的文件存放于类路径下面的 META-INF/services
文件夹中,并且文件中的每一行的内容都是一个实现类的全路径;ServiceLoader
:JDK
内置的一个类加载器,用于加载配置文件中的实现类;上面说了 SPI
的几个概念,接下来就通过一个栗子来带大家感受一下具体的用法。
创建一个接口,这里我们创建一个解压缩的接口,其中定义了压缩和解压的两个方法。
package com.example.demo.spi; /** * <br> * <b>Function:</b><br> * <b>Author:</b>@author ziyou<br> * <b>Date:</b>2022-10-08 21:31<br> * <b>Desc:</b>无<br> */ public interface Compresser { byte[] compress(byte[] bytes); byte[] decompress(byte[] bytes); }
再写两个对应的实现类,分别是 GzipCompresser.java
和 WinRarCompresser.java
代码如下
package com.example.demo.spi.impl; import com.example.demo.spi.Compresser; import java.nio.charset.StandardCharsets; /** * <br> * <b>Function:</b><br> * <b>Author:</b>@author ziyou<br> * <b>Date:</b>2022-10-08 21:33<br> * <b>Desc:</b>无<br> */ public class GzipCompresser implements Compresser { @Override public byte[] compress(byte[] bytes) { return"compress by Gzip".getBytes(StandardCharsets.UTF_8); } @Override public byte[] decompress(byte[] bytes) { return "decompress by Gzip".getBytes(StandardCharsets.UTF_8); } }
package com.example.demo.spi.impl; import com.example.demo.spi.Compresser; import java.nio.charset.StandardCharsets; /** * <br> * <b>Function:</b><br> * <b>Author:</b>@author ziyou<br> * <b>Date:</b>2022-10-08 21:33<br> * <b>Desc:</b>无<br> */ public class WinRarCompresser implements Compresser { @Override public byte[] compress(byte[] bytes) { return "compress by WinRar".getBytes(StandardCharsets.UTF_8); } @Override public byte[] decompress(byte[] bytes) { return "decompress by WinRar".getBytes(StandardCharsets.UTF_8); } }
创建配置文件,我们接着在 resources
目录下创建一个名为 META-INF/services
的文件夹,在其中创建一个名为 com.example.demo.spi.Compresser
的文件,其中的内容如下:
com.example.demo.spi.impl.WinRarCompresser com.example.demo.spi.impl.GzipCompresser
注意该文件的名称必须是接口的全路径,文件里面的内容每一行都是一个实现类的全路径,多个实现类就写在多行里面,效果如下。
有了上面的接口,实现类和配置文件,接下来我们就可以使用 ServiceLoader
动态加载实现类,来实现 SPI
技术了,如下所示:
package com.example.demo; import com.example.demo.spi.Compresser; import java.nio.charset.StandardCharsets; import java.util.ServiceLoader; public class TestSPI { public static void main(String[] args) { ServiceLoader<Compresser> compressers = ServiceLoader.load(Compresser.class); for (Compresser compresser : compressers) { System.out.println(compresser.getClass()); } } }
可以看到我们正常的获取到了接口的实现类,并且可以直接使用实现类的解压缩方法。
知道了如何使用 SPI
接下来我们来研究一下是如何实现的,通过上面的测试我们可以看到,核心的逻辑是 ServiceLoader.load()
方法,这个方法有点类似于 Spring
中的根据接口获取所有实现类一样。
点开 ServiceLoader
我们可以看到有一个常量 PREFIX
,如下所示,这也是为什么我们必须在这个路径下面创建配置文件,因为JDK
代码里面会从这个路径里面去读取我们的文件。
标签:Java,系统,操作系统,硬件配置,安装,电子计算机,存储器,主机 来源:
本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享; 2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关; 3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关; 4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除; 5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。