SPI的全名為Service Provider Interface.普通開發人員可能不熟悉,因為這個是針對廠商或者插件的。在java.util.ServiceLoader的文檔裡有比較詳細的介紹。究其思想,其實是和”Callback”差不多。“Callback”的思想是在我們調用API的時候,我們可以自己寫一段邏輯代碼,傳入到API裡面,API內部在合適的時候會調用它,從而實現某種程度的“定制”。
典型的是Collections.sort(List<T> list,Comparator<? super T> c)這個方法,它的第二個參數是一個實現Comparator接口的實例。我們可以根據自己的排序規則寫一個類,實現此接口,傳入此方法,那麼這個方法就會根據我們的規則對list進行排序。
把這個思想擴展開來,我們用SPI來重新實現上面的例子。客戶把自己的排序規則寫成一個類,並且打包成Jar文件,這個Jar文件裡面必須有META-INF目錄,其下又有services目錄,其下有一個文本文件,文件名即為接口的全名:java.util.Comparator。
–META-INF
–services
–java.util.Comparator
文件內容隻有一行:
com.company1.ComparatorProvider
這一行是你實現瞭Comparator接口的類的全名,它的代碼如下:
package com.company1;
import java.util.Comparator;
import com.mycompany.myapp.MyItem;
public class ComparatorProvider implements Comparator<MyItem>{
@Override
public int compare(MyItem o1, MyItem o2) {
//依據name排序
return o1.getName().compareTo(o2.getName());
}
}
編譯打包後,把它放到你的主程序的class path裡。下面是你的主程序:
//從class path中所有Jar的META-INF目錄中搜索,找到合適的類並加載。
private static ServiceLoader<Comparator> serviceLoader
= ServiceLoader.load(Comparator.class);
public static void main(String[] args)
{
List<MyItem> myList = new ArrayList<MyItem>();
myList.add(new MyItem(2,”c”,”hhh”));
myList.add(new MyItem(3,”k”,”ooo”));
myList.add(new MyItem(4,”d”,”ppp”));
myList.add(new MyItem(5,”b”,”ggg”));
showList(myList);
Collections.sort(myList,getCompartor());
showList(myList);
}
@SuppressWarnings(“unchecked”)
private static Comparator<MyItem> getCompartor() {
for(Comparator service : serviceLoader)
{
return (Comparator<MyItem>)service;
}
return null;
}
要註意的是serviceLoader開始隻是加載類,實例化要到第一次用的時候。類MyItem和方法showList並不重要,所以你不必在意。你可以按照這個規則,寫另外一個排序規則的Jar,隨時可以更換你的排序規則.