首先说一下问题。
昨日在看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