Java SPI机制
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
最后通过反射加载并创建对象即可,原理还是很简单的!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
WPF:拖动父窗口行为
原文 WPF:拖动父窗口行为 这次只是一个快速的帖子:当我点击并拖动特定的UIElement时,我需要能够重新定位WPF窗口。目的是重新创建在标准Windows标题栏上单击和拖动的行为(在我的情况下,我正在实现我自己的标题栏)。 事实证明这很容易实现,因此我将功能包装在一个简单的WPF行为中。您可以简单地将此行为附加到任何屏幕上的元素,它将自动找到父窗口,并将所有内容挂钩。 C# /// <summary> /// Attach this behaviour to any framework element to allow the entire parent window to be moved /// when you click and drag this element. /// </summary> public class DragWindowBehaviour : Behavior<FrameworkElement> { private Window _parentWindow; protected override void OnA...
-
下一篇
Java父子类加载顺序
加载顺序 先上桌结论: 父类静态属性(成员变量) > 父类静态代码块 > 子类静态属性 > 子类静态代码块 > 父类非静态属性 > 父类非静态代码块 > 父类构造器 > 子类非静态属性 > 子类非静态代码块 > 子类构造器 这么长怎么记呀?! 这里帮大家小结几个特点: 静态属性和代码块,当且仅当该类在程序中第一次被 new 或者第一次被类加载器调用时才会触发(不考虑永久代的回收)。也正是因为上述原因,类优先于对象 加载/new,即 静态优先于非静态。 属性(成员变量)优先于构造方法,可以这么理解,加载这整个类,需要先知道类具有哪些属性,并且这些属性初始化完毕之后,这个类的对象才算是完整的。另外,非静态代码块其实就是对象 new 的准备工作之一,算是一个不接受任何外来参数的构造方法。因此,属性 > 非静态代码块 > 构造方法。 有趣的是,静态部分(前4个)是父类 > 子类,而 非静态部分(后6个)也是父类 > 子类。 另外容易忽略的是,非静态代码块在每次 new 对象时都会运行,可以理解:非静态代码块是正式构造...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- CentOS关闭SELinux安全模块
- Windows10,CentOS7,CentOS8安装Nodejs环境
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS8编译安装MySQL8.0.19
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- MySQL数据库在高并发下的优化方案
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- Dcoker安装(在线仓库),最新的服务器搭配容器使用