由于在阅读dubbo源码的时候,发现了spi,这里做一下简单的记录。
一、spi的功能
为接口自动寻找实现类。
二、spi的实现
1、标准制定者制定接口
2、不同厂商编写针对于该接口的实现类,并在jar的“classpath:META-INF/services/全接口名称”文件中指定相应的实现类全类名
3、开发者直接引入相应的jar,就可以实现为接口自动寻找实现类的功能
三、java实现Demo
整体项目结构 
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.itcv</groupId>
<artifactId>spiDemo</artifactId>
<version>1.0-SNAPSHOT</version>
</project>
1、定义日志接口
package com.itcv.spi.log;
public interface Log {
public void excute();
}
2、定义日志接口实现
package com.itcv.spi.log.log4j;
import com.itcv.spi.log.Log;
public class Log4j implements Log {
public void excute() {
System.out.println("this is Log4j.....");
}
}
package com.itcv.spi.log.logbak;
import com.itcv.spi.log.Log;
public class LogBak implements Log {
public void excute() {
System.out.println("this is LogBak 实现");
}
}
3、指定使用的实现文件:META-INF/services/com.itcv.spi.log.Log
com.itcv.spi.log.logbak.LogBak
注意 这里指定了实现类Logback,那么加载的时候就会自动为Log接口指定实现类为Logback。
这里也可以指定两个实现类,那么在实际中使用哪一个实现类,就需要使用额外的手段来控制。
4、加载实现主类
package com.itcv.spi.log;
import java.util.Iterator;
import java.util.ServiceLoader;
public class Main {
public static void main(String[] args) {
ServiceLoader<Log> serviceLoader = ServiceLoader.load(Log.class);
Iterator<Log> iterator = serviceLoader.iterator();
while (iterator.hasNext()) {
Log log = iterator.next();
log.excute();
}
}
}
注意:
ServiceLoader不是实例化以后,就去读取配置文件中的具体实现,并进行实例化。而是等到使用迭代器去遍历的时候,才会加载对应的配置文件去解析,调用hasNext方法的时候会去加载配置文件进行解析,调用next方法的时候进行实例化并缓存
- 具体见“源码分析”
四、源码解析
[源码解析](https://www.cnblogs.com/java-zhao/p/7617143.html)