首先交代下背景,何为Java SPI?
SPI 全称为 (Service Provider Interface) ,是JDK内置的一种服务提供发现机制。
为什么需要SPI?
我们的现代系统越来越庞大,如果设计架构有问题,就可能牵一发而动全身,在面向对象中我们推荐基于接口编程,模块之间基于接口编程,这样的好处显而易见,不在代码中进行硬编码,不同的实现者按照接口规范实现自己内部操作,然后在使用的时候再根据 SPI 的规范去获取对应的服务提供者的服务实现。通过 SPI 服务加载机制进行服务的注册和发现,可以有效的避免在代码中将服务提供者写死。从而可以基于接口编程,实现模块间的解耦。
SPI机制约定:
1.在 META-INF/services/ 目录中创建以接口全限定名命名的文件,该文件内容为API具体实 现类的全限定名
2.使用 ServiceLoader 类动态加载 META-INF 中的实现类
3.如 SPI 的实现类为 Jar 则需要放在主程序 ClassPath 中
4.API 具体实现类必须有一个不带参数的构造方法
如图:
现在已经被使用的案例:
- common-logging Apache最早提供的日志的门面接口。只有接口,没有实现。具体方 案由各提供商实现, 发现日志提供商是通过扫描 META- INF/services/org.apache.commons.logging.LogFactory配置文件,通过读取该文 件的内容 找到日志提工商实现类。只要我们的日志实现里包含了这个文件,并在文件里 制定LogFactory工厂接口的实现类即可。
- JDBC jdbc4.0以前, 开发人员还需要基于Class.forName("xxx")的方式来装载驱动。 创建连接: DriverManage.getConnection()中,有Connection con = aDriver.driver.connect(url, info); driver成员变量,是java.sql.Driver接口,Java对外公开的一个加载驱动接口,Java并未实现,至于实现这个接口由各个Jdbc厂商去实现。 如MySQL,mysql-connector-java-5.1.38.jar包下面META-INF.services包下有个java.sql.Driver文件打开文件有下面两行 com.mysql.jdbc.Driver com.mysql.fabric.jdbc.FabricMySQLDriver
示例Demo:
创建maven项目
目录如下:
代码
OrderService.java
package com.demo.spi.service;
public interface OrderService {
int getOrderCountById(int id);
}
CustomerOrderServiceImpl.java
package com.demo.spi.impl;
import com.demo.spi.service.OrderService;
public class CustomerOrderServiceImpl implements OrderService {
public int getOrderCountById(int id) {
System.out.println("cutomer order count is 10");
return 10;
}
}
AgencyOrderServiceImpl.java
package com.demo.spi.impl;
import com.demo.spi.service.OrderService;
public class AgencyOrderServiceImpl implements OrderService {
public int getOrderCountById(int id) {
System.out.println("agency order count is 20");
return 20;
}
}
META-INF下文件名:com.demo.spi.service.OrderService,文件内容:
com.demo.spi.impl.AgencyOrderServiceImpl
com.demo.spi.impl.CustomerOrderServiceImpl复制代码
4.新建测试项目java project
运行main方法之前我们需要将第一个项目进行打包 jar 依赖到第二个java 项目中来,完成之后点击run,可以看到打印出了,我们在项目1 中的两个serviceImpl方法的输出,也就是说ServiceLoader 动态的通过jar中的配置找到了 项目1中的实现类并且把他记载到了内存中,我们就可以直接调用项目1中提供的两个实现类,并且正确输出。
最后
顺便在此给大家推荐一个Java方面的交流学习群:957734884,里面会分享一些高级面试题,还有资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化这些成为架构师必备的知识体系,主要针对Java开发人员提升自己,突破瓶颈,相信你来学习,会有提升和收获。在这个群里会有你需要的内容 朋友们请抓紧时间加入进来吧
原文地址:https://blog.51cto.com/14207399/2353202