0
很久没有写博客了,这一两个月基本上就是找工作——离职——入职。到今天基本上入职近一个月了,在看公司代码的时候发现使用到了SPI技术来进行解耦,这篇文章主要写下SPI的基本使用。
1. SPI基本概念
SPI的全名叫Service Provider Interface,其主要是针对厂商或者插件的。我的理解就是上层提供接口,我们需要去实现,并且上层只需要根据我们的配置文件即可拿到我们的实现类(反射获取)。
2. SPI代码
先看下目录结构:
2.1 接口定义
定义一个接口IStudy
:
public interface IStudy {
void study();
}
2.2 编写实现类
编写两个实现类NormalStudy
和SuperStudy
:
public class NormalStudy implements IStudy {
@Override
public void study() {
System.out.println("normal study");
}
}
public class SuperStudy implements IStudy {
@Override
public void study() {
System.out.println("super study");
}
}
2.3 创建com.nick.test.IStudy文件
在src目录下创建META-INF\services\com.nick.test.IStudy
文件,并在文件中添加两个实现类:
com.nick.test.NormalStudy
com.nick.test.SuperStudy
2.4 测试
public class Test {
public static void main(String[] args) {
ServiceLoader<IStudy> searches = ServiceLoader.load(IStudy.class);
Iterator<IStudy> iterator = searches.iterator();
while (iterator.hasNext()) {
iterator.next().study();
}
}
}
2.5 输出
3 原理简单分析
大致原理其实通过上面的示例应该都知道了:通过读取META-INF/services/com.nick.test.IStudy
文件的内容来确定接口对应的实现类,最终通过反射来进行实现类的加载。
核心代码:
try {
Enumeration<URL> resource = Test.class.getClassLoader().getResources("META-INF/services/" + IStudy.class.getName());
while (resource.hasMoreElements()) {
URL url = resource.nextElement();
InputStream inputStream = url.openStream();
BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream, "utf-8"));
String s;
while ((s = bufferedReader.readLine()) != null) {
System.out.println(s);
}
}
} catch (IOException e) {
e.printStackTrace();
}
输出:
com.nick.test.NormalStudy
com.nick.test.SuperStudy
最后通过反射加载并创建对象即可,原理还是很简单的!