2025-02-10

 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,隨時可以更換你的排序規則.

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *