ServiceLoader主要的功能是用來完成對SPI的provider的加載。數組
先看下它的成員:緩存
1 public final class ServiceLoader<S> 2 implements Iterable<S> { 3 4 private static final String PREFIX = "META-INF/services/"; 5 6 // The class or interface representing the service being loaded 7 private final Class<S> service; 8 9 // The class loader used to locate, load, and instantiate providers 10 private final ClassLoader loader; 11 12 // The access control context taken when the ServiceLoader is created 13 private final AccessControlContext acc; 14 15 // Cached providers, in instantiation order 16 private LinkedHashMap<String,S> providers = new LinkedHashMap<>(); 17 18 // The current lazy-lookup iterator 19 private LazyIterator lookupIterator; 20 21 ...... 22 23 }
能夠看到他首先是實現了Iterable接口,能夠迭代。
PREFIX:指明瞭路徑是在"META-INF/services/"下。
service:表示正在加載的服務的類或接口。
loader:使用的類加載器。
acc:建立ServiceLoader時獲取的訪問控制上下文。
providers :緩存的服務提供集合。
lookupIterator:是其內部使用的迭代器,用於類的懶加載,只有在迭代時加載。安全
其構造方法是一個private方法,不對外提供,在使用時咱們須要調用其靜態的load方法,由其自身產生ServiceLoader對象:app
1 public static <S> ServiceLoader<S> load(Class<S> service) { 2 ClassLoader cl = Thread.currentThread().getContextClassLoader(); 3 return ServiceLoader.load(service, cl); 4 } 5 6 public static <S> ServiceLoader<S> load(Class<S> service, 7 ClassLoader loader) { 8 return new ServiceLoader<>(service, loader); 9 }
能夠看到對load方法進行了重載,其中參數service是要加載的類;單參方法沒有類加載器,使用的是當前線程的類加載器;最後調用的是雙參的load方法;而雙參的load方法也很簡單,只是直接調用ServiceLoader的構造方法,實例化了一個對象。ide
1 private ServiceLoader(Class<S> svc, ClassLoader cl) { 2 service = Objects.requireNonNull(svc, "Service interface cannot be null"); 3 loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl; 4 acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null; 5 reload(); 6 }
能夠看到其構造方法邏輯依舊很簡單,首先是判斷傳入的svc(即傳入的service)是否爲空,如果爲空直接報異常,不然給service 成員賦值:源碼分析
1 public static <T> T requireNonNull(T obj, String message) { 2 if (obj == null) 3 throw new NullPointerException(message); 4 return obj; 5 }
而後給進行cl的非空判斷,給loader 成員賦值;接着給acc 成員賦值,其根據是否設置了安全管理器SecurityManager來賦值;最後調用reload方法。ui
1 public void reload() { 2 providers.clear(); 3 lookupIterator = new LazyIterator(service, loader); 4 }
能夠看到reload方法是一個public方法,那麼在每次調用reload時就須要將以前加載的清空掉,因此直接使用providers這個map的clear方法清空掉緩存;接着使用剛纔賦值後的service和loader產生一個LazyIterator對象賦值給lookupIterator成員。this
LazyIterator是ServiceLoader的內部類,其定義以下:spa
1 private class LazyIterator 2 implements Iterator<S> { 3 Class<S> service; 4 ClassLoader loader; 5 Enumeration<URL> configs = null; 6 Iterator<String> pending = null; 7 String nextName = null; 8 9 private LazyIterator(Class<S> service, ClassLoader loader) { 10 this.service = service; 11 this.loader = loader; 12 } 13 ...... 14 }
這裏就能夠看到ServiceLoader的實際加載過程就交給了LazyIterator來作,將ServiceLoader的service和loader成員分別賦值給了LazyIterator的service和loader成員。
configs是服務的URL枚舉;
pending是保存要加載的服務的名稱集合;
nextName是下一個要加載的服務名稱;線程
ServiceLoader實現了Iterable接口,其實現的iterator方法以下:
1 public Iterator<S> iterator() { 2 return new Iterator<S>() { 3 Iterator<Map.Entry<String,S>> knownProviders 4 = providers.entrySet().iterator(); 5 6 public boolean hasNext() { 7 if (knownProviders.hasNext()) 8 return true; 9 return lookupIterator.hasNext(); 10 } 11 12 public S next() { 13 if (knownProviders.hasNext()) 14 return knownProviders.next().getValue(); 15 return lookupIterator.next(); 16 } 17 18 public void remove() { 19 throw new UnsupportedOperationException(); 20 } 21 22 }; 23 }
能夠看到它是直接建立了一個Iterator對象返回;其knownProviders成員直接獲取providers的entrySet集合的迭代器;在hasNext和next方法中咱們能夠看到,它是先經過判斷knownProviders裏有沒有(即providers),若沒有再去lookupIterator中找;
前面咱們能夠看到providers裏並沒用put任何東西,那麼就說明put操做也是在lookupIterator中完成的。
先看到lookupIterator的next方法:
1 public S next() { 2 if (acc == null) { 3 return nextService(); 4 } else { 5 PrivilegedAction<S> action = new PrivilegedAction<S>() { 6 public S run() { return nextService(); } 7 }; 8 return AccessController.doPrivileged(action, acc); 9 } 10 }
首先根據判斷acc是否爲空,若爲空則說明沒有設置安全策略直接調用nextService方法,不然以特權方式調用nextService方法。
1 private S nextService() { 2 if (!hasNextService()) 3 throw new NoSuchElementException(); 4 String cn = nextName; 5 nextName = null; 6 Class<?> c = null; 7 try { 8 c = Class.forName(cn, false, loader); 9 } catch (ClassNotFoundException x) { 10 fail(service, 11 "Provider " + cn + " not found"); 12 } 13 if (!service.isAssignableFrom(c)) { 14 fail(service, 15 "Provider " + cn + " not a subtype"); 16 } 17 try { 18 S p = service.cast(c.newInstance()); 19 providers.put(cn, p); 20 return p; 21 } catch (Throwable x) { 22 fail(service, 23 "Provider " + cn + " could not be instantiated", 24 x); 25 } 26 throw new Error(); // This cannot happen 27 }
首先根據hasNextService方法判斷,若爲false直接拋出NoSuchElementException異常,不然繼續執行。
hasNextService方法:
1 private boolean hasNextService() { 2 if (nextName != null) { 3 return true; 4 } 5 if (configs == null) { 6 try { 7 String fullName = PREFIX + service.getName(); 8 if (loader == null) 9 configs = ClassLoader.getSystemResources(fullName); 10 else 11 configs = loader.getResources(fullName); 12 } catch (IOException x) { 13 fail(service, "Error locating configuration files", x); 14 } 15 } 16 while ((pending == null) || !pending.hasNext()) { 17 if (!configs.hasMoreElements()) { 18 return false; 19 } 20 pending = parse(service, configs.nextElement()); 21 } 22 nextName = pending.next(); 23 return true; 24 }
hasNextService方法首先根據nextName成員是否爲空判斷,若不爲空,則說明已經初始化過了,直接返回true,不然繼續執行。接着configs成員是否爲空,configs 是一個URL的枚舉,如果configs 沒有初始化,就須要對configs初始化。
configs初始化邏輯也很簡單,首先根據PREFIX前綴加上PREFIX的全名獲得完整路徑,再根據loader的有無,獲取URL的枚舉。其中fail方法時ServiceLoader的靜態方法,用於異常的處理,後面給出。
在configs初始化完成後,還須要完成pending的初始化或者添加。
能夠看到只有當pending爲null,或者沒有元素時才進行循環。循環時如果configs裏沒有元素,則直接返回false;不然調用ServiceLoader的parse方法,經過service和URL給pending賦值;
parse方法:
1 private Iterator<String> parse(Class<?> service, URL u) 2 throws ServiceConfigurationError { 3 InputStream in = null; 4 BufferedReader r = null; 5 ArrayList<String> names = new ArrayList<>(); 6 try { 7 in = u.openStream(); 8 r = new BufferedReader(new InputStreamReader(in, "utf-8")); 9 int lc = 1; 10 while ((lc = parseLine(service, u, r, lc, names)) >= 0); 11 } catch (IOException x) { 12 fail(service, "Error reading configuration file", x); 13 } finally { 14 try { 15 if (r != null) r.close(); 16 if (in != null) in.close(); 17 } catch (IOException y) { 18 fail(service, "Error closing configuration file", y); 19 } 20 } 21 return names.iterator(); 22 }
能夠看到parse方法直接經過URL打開輸入流,經過parseLine一行一行地讀取將結果保存在names數組裏。
parseLine方法:
1 private int parseLine(Class<?> service, URL u, BufferedReader r, int lc, 2 List<String> names) 3 throws IOException, ServiceConfigurationError { 4 String ln = r.readLine(); 5 if (ln == null) { 6 return -1; 7 } 8 int ci = ln.indexOf('#'); 9 if (ci >= 0) ln = ln.substring(0, ci); 10 ln = ln.trim(); 11 int n = ln.length(); 12 if (n != 0) { 13 if ((ln.indexOf(' ') >= 0) || (ln.indexOf('\t') >= 0)) 14 fail(service, u, lc, "Illegal configuration-file syntax"); 15 int cp = ln.codePointAt(0); 16 if (!Character.isJavaIdentifierStart(cp)) 17 fail(service, u, lc, "Illegal provider-class name: " + ln); 18 for (int i = Character.charCount(cp); i < n; i += Character.charCount(cp)) { 19 cp = ln.codePointAt(i); 20 if (!Character.isJavaIdentifierPart(cp) && (cp != '.')) 21 fail(service, u, lc, "Illegal provider-class name: " + ln); 22 } 23 if (!providers.containsKey(ln) && !names.contains(ln)) 24 names.add(ln); 25 } 26 return lc + 1; 27 }
parseLine方法就是讀該URL對應地文件地一行,能夠看到經過對「#」的位置判斷,忽略註釋,而且剔除空格,接着是一系列的參數合法檢驗,而後判斷providers和names裏是否都沒包含這個服務名稱,若都沒包含names直接add,最後返回下一行的行標;
當parse將全部內容讀取完畢,返回names.iterator()賦值給hasNextService中的pending。循環結束,獲取pending中的第一個元素賦值給nextName,返回true,hasNextService方法結束。
在nextService方法往下執行時,先用cn保存nextName的值,再讓nextName=null,爲下一次的遍歷作準備;接着經過類加載,加載名爲cn的類,再經過該類實例化對象,並用providers緩存起來,最後返回該實例對象。
其中cast方法是判斷對象是否合法:
1 public T cast(Object obj) { 2 if (obj != null && !isInstance(obj)) 3 throw new ClassCastException(cannotCastMsg(obj)); 4 return (T) obj; 5 }
至此ServiceLoader的迭代器的next方法結束。其hasNext方法與其相似,就不詳細分析了。
而其remove方法就更直接,直接拋出異常來避免可能出現的危險狀況:
1 public void remove() { 2 throw new UnsupportedOperationException(); 3 }
其中使用到的靜態fail方法只是拋出異常:
1 private static void fail(Class<?> service, String msg, Throwable cause) 2 throws ServiceConfigurationError { 3 throw new ServiceConfigurationError(service.getName() + ": " + msg, 4 cause); 5 } 6 7 private static void fail(Class<?> service, String msg) 8 throws ServiceConfigurationError { 9 throw new ServiceConfigurationError(service.getName() + ": " + msg); 10 } 11 12 private static void fail(Class<?> service, URL u, int line, String msg) 13 throws ServiceConfigurationError { 14 fail(service, u + ":" + line + ": " + msg); 15 }
ServiceLoader除了load的兩個方法外還有個loadInstalled方法:
1 public static <S> ServiceLoader<S> loadInstalled(Class<S> service) { 2 ClassLoader cl = ClassLoader.getSystemClassLoader(); 3 ClassLoader prev = null; 4 while (cl != null) { 5 prev = cl; 6 cl = cl.getParent(); 7 } 8 return ServiceLoader.load(service, prev); 9 }
該方法與load方法不一樣在於loadInstalled使用的是擴展類加載器,而load使用的是傳入進來的或者是線程的上下文類加載器,其餘都同樣。
ServiceLoader源碼分析到此所有結束。