概念
Service Provider Interface
规则
- 在resource/META-INF/services 创建一个以接口全限定名为命名的文件,内容写上实现类的全限定名
- 接口实现类在classpath路径下
- 主程序通过 java.util.ServiceLoader 动态装载实现模块(扫描META-INF/services目录下的配置文件找到实现类,装载到 JVM)
好处
解耦,主程序和实现类之间不用硬编码
例子
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-16 16:14
*/
public interface SPIService {
void execute();
}
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-16 16:16
*/
public class SpiImpl1 implements SPIService {
@Override
public void execute() {
System.out.println("SpiImpl1.execute()");
}
}
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-16 16:16
*/
public class SpiImpl2 implements SPIService {
@Override
public void execute() {
System.out.println("SpiImpl2.execute()");
}
}
在 resources/META-INF/services/目录下创建文件名为com.mousycoder.mycode.thinkinginjvm.SPIService的文件,内容com.mousycoder.mycode.thinkinginjvm.SpiImpl1com.mousycoder.mycode.thinkinginjvm.SpiImpl2
主程序
package com.mousycoder.mycode.thinking_in_jvm;
import sun.misc.Service;
import java.util.Iterator;
import java.util.ServiceLoader;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-16 16:21
*/
public class SPIMain {
public static void main(String[] args) {
Iterator<SPIService> providers = Service.providers(SPIService.class);
ServiceLoader<SPIService> load = ServiceLoader.load(SPIService.class);
while (providers.hasNext()){
SPIService ser = providers.next();
ser.execute();
}
System.out.println("-----------------------");
Iterator<SPIService> iterator = load.iterator();
while (iterator.hasNext()){
SPIService ser = iterator.next();
ser.execute();
}
}
}
输出
SpiImpl1.execute()
SpiImpl2.execute()
-----------------------
SpiImpl1.execute()
SpiImpl2.execute()
本文详细介绍了Java SPI(Service Provider Interface)机制的工作原理及应用实例。SPI允许在不修改主程序代码的情况下,通过配置文件动态加载和使用不同的实现模块,有效实现了解耦。
593

被折叠的 条评论
为什么被折叠?



