日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解java實踐SPI機制及淺析源碼

瀏覽:15日期:2022-08-28 14:13:47

1.概念

正式步入今天的核心內容之前,溪源先給大家介紹一下關于SPI機制的相關概念,最后會提供實踐源代碼。

SPI即Service Provider Interface,屬于JDK內置的一種動態的服務提供發現機制,可以理解為運行時動態加載接口的實現類。更甚至,大家可以將SPI機制與設計模式中的策略模式建立聯系。

SPI機制:

詳解java實踐SPI機制及淺析源碼

從上圖中理解SPI機制:標準化接口+策略模式+配置文件;

SPI機制核心思想:系統設計的各個抽象,往往有很多不同的實現方案,在面向的對象的設計里,一般推薦模塊之間基于接口編程,模塊之間不對實現類進行硬編碼。一旦代碼里涉及具體的實現類,就違反了可拔插的原則,如果需要替換一種實現,就需要修改代碼。為了實現在模塊裝配的時候能不在程序里動態指明,這就需要一種服務發現機制

使用場景:

1.數據庫驅動加載:面對不同廠商的數據庫,JDBC需要加載不同類型的數據庫驅動; 2.日志接口實現:SLF4J加載不同日志實現類; 3.溪源在實際開發中也使用了SPI機制:面對不同儀器平臺的結果文件上傳需要解析具體的結果,文件不同,解析邏輯不同,因此采用SPI機制能夠解耦和降低維護成本;

SPI機制使用約定:

從上面的圖中,我們可以清晰的知道SPI的三部分:接口+實現類+配置文件;因此,項目中若要利用SPI機制,則需要遵循以下約定:

當服務提供者提供了接口的一種具體實現后,在jar包的META-INF/services目錄下創建一個以“接口全限定名”為命名的文件,內容為實現類的全限定名。 主程序通過java.util.ServiceLoder動態裝載實現模塊,它通過掃描META-INF/services目錄下的配置文件找到實現類的全限定名,把類加載到JVM;

注意:除SPI,我還發布了最新Java架構項目實戰教程+大廠面試題庫, 點擊此處免費獲取,小白勿進!

2.實踐

整體包結構如圖:

詳解java實踐SPI機制及淺析源碼

新建標準化接口:

public interface SayService { void say(String word);}

建立兩個實現類

@Servicepublic class ASayServiceImpl implements SayService { @Override public void say(String word) { System.out.println(word + ' A say: I am a boy'); }}@Servicepublic class BSayServiceImpl implements SayService { @Override public void say(String word) { System.out.println(word + ' B say: I am a girl'); }}

新建META-INF/services目錄和配置文件(以接口全限定名)

配置文件內容為實現類全限定名

com.qxy.spi.impl.ASayServiceImplcom.qxy.spi.impl.BSayServiceImpl

單測

@SpringBootTest@RunWith(SpringRunner.class)public class SpiTest { static ServiceLoader<SayService> services = ServiceLoader.load(SayService.class); @Test public void test1() { for (SayService sayService : services) { sayService.say('Hello'); } }}

結果

Hello A say: I am a boyHello B say: I am a girl

3.源碼

源碼主要加載流程如下:

應用程序調用ServiceLoader.load方法 ServiceLoader.load方法內先創建一個新的ServiceLoader,并實例化該類中的成員變量;

loader(ClassLoader類型,類加載器) acc(AccessControlContext類型,訪問控制器) providers(LinkedHashMap<String,S>類型,用于緩存加載成功的類) lookupIterator(實現迭代器功能)

應用程序通過迭代器接口獲取對象實例 ServiceLoader先判斷成員變量providers對象中(LinkedHashMap<String,S>類型)是否有緩存實例對象,如果有緩存,直接返回。如果沒有緩存,執行類的裝載。

讀取META-INF/services/下的配置文件,獲得所有能被實例化的類的名稱,值得注意的是,ServiceLoader可以跨越jar包獲取META-INF下的配置文件; 通過反射方法Class.forName()加載類對象,并用instance()方法將類實例化。 把實例化后的類緩存到providers對象中,(LinkedHashMap<String,S>類型) 然后返回實例對象。

public final class ServiceLoader<S> implements Iterable<S>{ // 加載具體實現類信息的前綴 private static final String PREFIX = 'META-INF/services/'; // 需要加載的接口 // The class or interface representing the service being loaded private final Class<S> service; // 用于加載的類加載器 // The class loader used to locate, load, and instantiate providers private final ClassLoader loader; // 創建ServiceLoader時采用的訪問控制上下文 // The access control context taken when the ServiceLoader is created private final AccessControlContext acc; // 用于緩存已經加載的接口實現類,其中key為實現類的完整類名 // Cached providers, in instantiation order private LinkedHashMap<String,S> providers = new LinkedHashMap<>(); // 用于延遲加載接口的實現類 // The current lazy-lookup iterator private LazyIterator lookupIterator; public void reload() { providers.clear(); lookupIterator = new LazyIterator(service, loader); } private ServiceLoader(Class<S> svc, ClassLoader cl) { service = Objects.requireNonNull(svc, 'Service interface cannot be null'); loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl; acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null; reload(); } private static void fail(Class<?> service, String msg, Throwable cause) throws ServiceConfigurationError { throw new ServiceConfigurationError(service.getName() + ': ' + msg, cause); } private static void fail(Class<?> service, String msg) throws ServiceConfigurationError { throw new ServiceConfigurationError(service.getName() + ': ' + msg); } private static void fail(Class<?> service, URL u, int line, String msg) throws ServiceConfigurationError { fail(service, u + ':' + line + ': ' + msg); } // Parse a single line from the given configuration file, adding the name // on the line to the names list. //具體解析資源文件中的每一行內容 private int parseLine(Class<?> service, URL u, BufferedReader r, int lc, List<String> names) throws IOException, ServiceConfigurationError { String ln = r.readLine(); if (ln == null) { //-1表示解析完成 return -1; } // 如果存在’#’字符,截取第一個’#’字符串之前的內容,’#’字符之后的屬于注釋內容 int ci = ln.indexOf(’#’); if (ci >= 0) ln = ln.substring(0, ci); ln = ln.trim(); int n = ln.length(); if (n != 0) { //不合法的標識:’ ’、’t’ if ((ln.indexOf(’ ’) >= 0) || (ln.indexOf(’t’) >= 0))fail(service, u, lc, 'Illegal configuration-file syntax'); int cp = ln.codePointAt(0); //判斷第一個 char 是否一個合法的 Java 起始標識符 if (!Character.isJavaIdentifierStart(cp))fail(service, u, lc, 'Illegal provider-class name: ' + ln); //判斷所有其他字符串是否屬于合法的Java標識符 for (int i = Character.charCount(cp); i < n; i += Character.charCount(cp)) {cp = ln.codePointAt(i);if (!Character.isJavaIdentifierPart(cp) && (cp != ’.’)) fail(service, u, lc, 'Illegal provider-class name: ' + ln); } //不存在則緩存 if (!providers.containsKey(ln) && !names.contains(ln))names.add(ln); } return lc + 1; } private Iterator<String> parse(Class<?> service, URL u) throws ServiceConfigurationError { InputStream in = null; BufferedReader r = null; ArrayList<String> names = new ArrayList<>(); try { in = u.openStream(); r = new BufferedReader(new InputStreamReader(in, 'utf-8')); int lc = 1; while ((lc = parseLine(service, u, r, lc, names)) >= 0); } catch (IOException x) { fail(service, 'Error reading configuration file', x); } finally { try {if (r != null) r.close();if (in != null) in.close(); } catch (IOException y) {fail(service, 'Error closing configuration file', y); } } return names.iterator(); } // Private inner class implementing fully-lazy provider lookup // private class LazyIterator implements Iterator<S> { Class<S> service; ClassLoader loader; // 加載資源的URL集合 Enumeration<URL> configs = null; // 需加載的實現類的全限定類名的集合 Iterator<String> pending = null; // 下一個需要加載的實現類的全限定類名 String nextName = null; private LazyIterator(Class<S> service, ClassLoader loader) { this.service = service; this.loader = loader; } private boolean hasNextService() { if (nextName != null) {return true; } if (configs == null) {try {// 資源名稱,META-INF/services + 全限定名 String fullName = PREFIX + service.getName(); if (loader == null) configs = ClassLoader.getSystemResources(fullName); else configs = loader.getResources(fullName);} catch (IOException x) { fail(service, 'Error locating configuration files', x);} } // 從資源中解析出需要加載的所有實現類的全限定名 while ((pending == null) || !pending.hasNext()) {if (!configs.hasMoreElements()) { return false;}pending = parse(service, configs.nextElement()); } //下一個需要加載的實現類全限定名 nextName = pending.next(); return true; } private S nextService() { if (!hasNextService())throw new NoSuchElementException(); String cn = nextName; nextName = null; Class<?> c = null; try { //反射構造Class實例c = Class.forName(cn, false, loader); } catch (ClassNotFoundException x) {fail(service, 'Provider ' + cn + ' not found'); } // 類型判斷,校驗實現類必須與當前加載的類/接口的關系是派生或相同,否則拋出異常終止 if (!service.isAssignableFrom(c)) {fail(service, 'Provider ' + cn + ' not a subtype'); } try { //強轉S p = service.cast(c.newInstance()); // 實例完成,添加緩存,Key:實現類全限定類名,Value:實現類實例providers.put(cn, p);return p; } catch (Throwable x) {fail(service, 'Provider ' + cn + ' could not be instantiated', x); } throw new Error(); // This cannot happen } public boolean hasNext() { if (acc == null) {return hasNextService(); } else {PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() { public Boolean run() { return hasNextService(); }};return AccessController.doPrivileged(action, acc); } } public S next() { if (acc == null) {return nextService(); } else {PrivilegedAction<S> action = new PrivilegedAction<S>() { public S run() { return nextService(); }};return AccessController.doPrivileged(action, acc); } } public void remove() { throw new UnsupportedOperationException(); } } public Iterator<S> iterator() { return new Iterator<S>() { Iterator<Map.Entry<String,S>> knownProviders= providers.entrySet().iterator(); public boolean hasNext() {if (knownProviders.hasNext()) return true;return lookupIterator.hasNext(); } public S next() {if (knownProviders.hasNext()) return knownProviders.next().getValue();return lookupIterator.next(); } public void remove() {throw new UnsupportedOperationException(); } }; } public static <S> ServiceLoader<S> load(Class<S> service, ClassLoader loader) { // 返回ServiceLoader的實例 return new ServiceLoader<>(service, loader); } public static <S> ServiceLoader<S> loadInstalled(Class<S> service) { ClassLoader cl = ClassLoader.getSystemClassLoader(); ClassLoader prev = null; while (cl != null) { prev = cl; cl = cl.getParent(); } return ServiceLoader.load(service, prev); } public String toString() { return 'java.util.ServiceLoader[' + service.getName() + ']'; }}

4.總結

SPI機制在實際開發中使用得場景也有很多。特別是統一標準的不同廠商實現,溪源也正是利用SPI機制(但略做改進,避免過多加載資源浪費)實現不同技術平臺的結果文件解析需求。

優點

使用Java SPI機制的優勢是實現解耦,使得第三方服務模塊的裝配控制的邏輯與調用者的業務代碼分離,而不是耦合在一起。應用程序可以根據實際業務情況啟用框架擴展或替換框架組件。

缺點

雖然ServiceLoader也算是使用的延遲加載,但是基本只能通過遍歷全部獲取,也就是接口的實現類全部加載并實例化一遍。如果你并不想用某些實現類,它也被加載并實例化了,這就造成了浪費。

源碼傳送門:SPI Service

到此這篇關于詳解java實踐SPI機制及淺析源碼的文章就介紹到這了,更多相關java SPI機制內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
成人羞羞在线观看网站| 老司机精品视频网| 成人台湾亚洲精品一区二区| 国产精品.xx视频.xxtv| 日本午夜精品一区二区三区电影| 日韩精品亚洲一区二区三区免费| 国产欧美午夜| 国产精品久久久久久模特 | 美女网站视频一区| 精品国产黄a∨片高清在线| 国产91欧美| 久久九九精品| 女同性一区二区三区人了人一| 日韩视频在线一区二区三区 | 一区二区精品伦理...| 亚洲成人一区在线观看| 久久成人一区| 国产精品白丝久久av网站| 久久精品色播| 国产日韩专区| 国产福利资源一区| 国产成人精品免费视| 黑丝一区二区三区| 日韩国产欧美在线播放| 黄色精品视频| 亚洲深夜福利| 久久精品国产99国产| 亚洲欧美高清| 成人一二三区| 亚洲欧美日韩国产| 日韩精品欧美成人高清一区二区| 国产在线视频欧美一区| 石原莉奈在线亚洲三区| 国产一区二区三区不卡视频网站| 最新亚洲激情| av资源亚洲| 日本不卡一区二区| 香蕉视频亚洲一级| 美女性感视频久久| 日av在线不卡| 九九综合九九| 精品日韩视频| 久久成人高清| 日韩av影院| 一区二区电影在线观看| 欧美精品羞羞答答| 中文字幕高清在线播放| 国产精品久久久久久久久久齐齐| 亚洲欧美日本日韩| 亚洲免费精品| 亚洲国产综合在线看不卡| 精品视频高潮| 国产精品地址| 欧美一区精品| 国产精品香蕉| 日本成人精品| 日韩1区2区3区| 在线免费观看亚洲| 首页国产欧美日韩丝袜| 免费视频一区三区| 嫩呦国产一区二区三区av| 中文字幕av一区二区三区人| 欧美女激情福利| 中文精品在线| 亚洲免费资源| 国产精品一区二区三区av| 中文字幕免费一区二区| 免费日韩一区二区| 亚洲精品成人一区| 国产精品亚洲二区| 亚洲电影有码| 日韩高清中文字幕一区二区| 999国产精品| 黄色成人91| 日韩av午夜在线观看| 欧产日产国产精品视频| 日韩精品一区二区三区免费观影 | 91精品亚洲| 丝袜亚洲精品中文字幕一区| 亚洲影院天堂中文av色| 日韩精品欧美精品| 久久一区视频| 国精品一区二区三区| 免费日韩视频| 国产精品对白| 中文亚洲欧美| 国产精品mv在线观看| se01亚洲视频 | 亚洲黄色免费看| 欧美不卡在线| 日本亚洲欧美天堂免费| 日韩美女国产精品| 日韩在线观看| 亚洲欧洲国产精品一区| 国产精品黑丝在线播放| 亚洲精品美女91| 亚洲最新无码中文字幕久久| 久久亚洲风情| 高清不卡亚洲| 久久精品99国产精品日本| 99精品视频在线| 美腿丝袜亚洲一区| 亚洲乱亚洲高清| 日韩美女一区二区三区在线观看| 日韩在线网址| 亚洲精品1区| 精品少妇av| 日本国产欧美| 午夜在线播放视频欧美| 日韩中文影院| 国产一区二区三区四区大秀| 日本国产欧美| 婷婷亚洲成人| 日本亚洲最大的色成网站www | 91精品尤物| 99pao成人国产永久免费视频 | 青青草国产精品亚洲专区无| 久久国产精品亚洲77777| 精品日韩毛片| 亚洲欧美日韩精品一区二区| 久久久夜夜夜| 亚洲国产专区| 亚州av乱码久久精品蜜桃| 首页国产精品| 久久久久欧美精品| 91精品一区二区三区综合| av中文字幕在线观看第一页| 久久精品免费看| 精品国产乱码久久久| 国产情侣久久| 久久只有精品| 97国产成人高清在线观看| 亚洲女同av| 自由日本语亚洲人高潮| 午夜在线精品| 国产极品一区| 韩国精品主播一区二区在线观看| 久久九九精品| 亚洲人成网77777色在线播放 | 水蜜桃精品av一区二区| 高清不卡亚洲| 极品裸体白嫩激情啪啪国产精品| 久久久久久美女精品| 亚洲一区观看| 欧美日本精品| 午夜影院一区| 亚洲18在线| 日韩亚洲一区在线| 一区二区亚洲视频| 久久精品国产久精国产爱| 久久免费黄色| 日本久久一区| 国产不卡人人| 日韩精品中文字幕一区二区| 国产一区丝袜| 水野朝阳av一区二区三区| 日韩成人在线看| 日韩影院二区| 久久国内精品自在自线400部| av高清一区| 麻豆精品视频在线| 亚洲资源网站| 欧洲毛片在线视频免费观看| 国产欧美激情| 久久av一区二区三区| 成人小电影网站| 91精品麻豆| 老司机久久99久久精品播放免费| 麻豆精品新av中文字幕| 日韩精品欧美精品| 模特精品在线| 亚洲美女久久精品| 国产精品久久久久久久久久妞妞| 免费成人在线影院| 香蕉精品视频在线观看| 精品一区二区三区在线观看视频| 亚洲精品九九| 另类激情亚洲| 日韩一区二区免费看| 日韩精品午夜| 性感美女一区二区在线观看| 日本黄色精品| 国产一区二区三区91| 国产欧美成人| 国产精品久久久久久模特| 国产精品一站二站| 婷婷综合福利| 色8久久久久| 日本不卡一区二区| 亚洲精品大片| 日本成人在线不卡视频| 日韩和欧美一区二区| 91欧美日韩在线| 国产精品15p| 久久久久久婷| 欧美~级网站不卡| 日韩亚洲精品在线| 在线看片日韩| 国产精品亚洲欧美| 精品美女视频 |