SPI实战
网上已经有很多介绍了SPI机制的文章,写得都非常棒。然而计算机是一门实战课程。最重要的是要把理论的知识转化为实战嘛。
SPI使用场景
略。实际中,如果真的用上,也是极少的。毕竟不常用。但是有一点可能很少有讲的。那就是SPI是支持双亲委派机制的。
言外之意就是,当普通用户在使用接口规范方的接口的时候,需要在pom包中依赖一些实现接口规范的厂商的jar包,这个jar包可以单独存在于
- $JAVA_HOME/jre/lib【boot加载器加载的目录】
- $JAVA_HOME/jre/lib/ext 【扩展类加载器加载jar包地址】,这里要注意,要把接口也一并放在/ext中【把接口jar直接放在lib中是无效的,可以自行测试】
项目设计方案
模块调用图 整体的模块设计是这样的。我们讲一下每个模块。
- spione 是由接口规范方提供的一个模块,其实质是定义了一系列行为【接口】。当然在其内部模拟一个DriverManager来启动驱动。实际代码
javapackage org.example;
import java.util.ServiceLoader;
public class DriverOne {
public static SPIFirst first;
static {
ServiceLoader<SPIFirst> load = ServiceLoader.load(SPIFirst.class);
int count = 0;
for (SPIFirst spiFirst : load) {
// 查找第一个
count++;
first = spiFirst;
break;
}
if(count == 0) {
throw new RuntimeException("你需要加载一个spi 容器: org.example.SPIFirst");
}
}
public String say(String sayName) {
return first.name(sayName);
}
}
-
spioneisp1是厂商根据接口规范方定义好的实现。用来提供用户使用的jar包。当然 厂商需要根据三部曲来实现自己
a) 实现接口
b) resource资源配置 -
spiuser 就是使用接口的用户了。用户只要在自己项目配置好pom文件,在pom文件中添加接口 jar包和 厂商1提供的jar包。
注意: 在调用的时候,不直接于使用厂商。所以我在图中用虚线来表示。
这样就使用java提供的spi能力来处理我们实际场景中问题。
当然,类似的springboot、dubbo等框架也实现了自己的spi能力,该能力有别于java提供的spi。
希望通过本篇能够帮助你进一步理解spi的能力。
这里有个问题:
如果把接口规范jar包放在 boot目录中, 而把厂商jar包放在 $JAVA_HOME/jre/lib/ext 【扩展类加载器加载jar包地址】。会发生什么问题?也就是使用场景中的第二点中提到的问题。欢迎道友评论哈。