Java的SPI机制心得

首先说一下问题。

昨日在看JDBC源码当看到DriverManage.getConnection()这个方法,点进去DriverManage类看到getConnection()方法里核心语句确实下面这个for(DriverInfo aDriver : registeredDrivers) {
Connection con = aDriver.driver.connect(url, info);
其中driver是DriverInfo类中的的Driver类型的成员变量,而Driver(java.sql.Driver)是个接口,所以点进去的connect方法只是个Driver接口里面定义的方法,并没有实现类,实现类哪里去了???(虽然后来在com.mysql.fabric.jdbc包里碰巧找到了,但不知道原理咋回事)。

今天在碰巧别人提了一下Spi机制,瞬间觉得和自己遇到的问题完全符合,小小的研究了一下自己总结理解如下:

1   为什么Driver是个接口:

      java.sql.Driver接口是Java对外公开的一个加载驱动接口,Java并未实现,至于实现这个接口由各个Jdbc厂商去实现就行了,好处是解藕,使得更具有灵活性,当然这也是面向对象的好处之一。

换句话说J2EE仅仅是一个标准,只是一个架构。真正的实现是不同提供商提供的。

2   怎么找到相应实现类:在mysql-connector-java-5.1.38.jar包下面META-INF.services包下有个java.sql.Driver文件打开文件有下面两行

com.mysql.jdbc.Driver

com.mysql.fabric.jdbc.FabricMySQLDriver

其中com.mysql.jdbc.Driver类是实现了java.sql.Driver接口,下面那com.mysql.fabric.jdbc.FabricMySQLDriver才是真正的实现类,

具体流程如下图:

3原理是什么:

当服务的提供者,提供了服务接口的一种实现之后,在jar包的META-INF/services/目录里同时创建一个以服务接口命名的文件。该文 件里就是实现该服务接口的具体实现类。而当外部程序装配这个

模块的时候,就能通过该jar包META-INF/services/里的配置文件找到具体的 实现类名,并装载实例化,完成模块的注入。 

基于这样一个约定就能很好的找到服务接口的实现类,而不需要再代码里制定。

jdk提供服务实现查找的一个工具类:java.util.ServiceLoader

posted @ 2018-01-19 09:54  王小森#  阅读(182)  评论(0编辑  收藏  举报