Java SPI机制

0

很久没有写博客了,这一两个月基本上就是找工作——离职——入职。到今天基本上入职近一个月了,在看公司代码的时候发现使用到了SPI技术来进行解耦,这篇文章主要写下SPI的基本使用。

1. SPI基本概念

SPI的全名叫Service Provider Interface,其主要是针对厂商或者插件的。我的理解就是上层提供接口,我们需要去实现,并且上层只需要根据我们的配置文件即可拿到我们的实现类(反射获取)。

2. SPI代码

先看下目录结构:


目录结构

2.1 接口定义

定义一个接口IStudy

public interface IStudy {
    void study();
}

2.2 编写实现类

编写两个实现类NormalStudySuperStudy

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

最后通过反射加载并创建对象即可,原理还是很简单的!

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容