首先說一下問題。

昨日在看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