电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

深入理解Java中的SPI機制

瀏覽:121日期:2022-08-17 08:20:04

本文通過探析JDK提供的,在開源項目中比較常用的Java SPI機制,希望給大家在實際開發實踐、學習開源項目提供參考。

1 SPI是什么

SPI全稱Service Provider Interface,是Java提供的一套用來被第三方實現或者擴展的API,它可以用來啟用框架擴展和替換組件。

整體機制圖如下:

深入理解Java中的SPI機制

Java SPI 實際上是“基于接口的編程+策略模式+配置文件”組合實現的動態加載機制。

系統設計的各個抽象,往往有很多不同的實現方案,在面向的對象的設計里,一般推薦模塊之間基于接口編程,模塊之間不對實現類進行硬編碼。一旦代碼里涉及具體的實現類,就違反了可拔插的原則,如果需要替換一種實現,就需要修改代碼。為了實現在模塊裝配的時候能不在程序里動態指明,這就需要一種服務發現機制。Java SPI就是提供這樣的一個機制:為某個接口尋找服務實現的機制。有點類似IOC的思想,就是將裝配的控制權移到程序之外,在模塊化設計中這個機制尤其重要。所以SPI的核心思想就是解耦。

2 使用場景

概括地說,適用于:調用者根據實際使用需要,啟用、擴展、或者替換框架的實現策略

比較常見的例子:

數據庫驅動加載接口實現類的加載

JDBC加載不同類型數據庫的驅動

日志門面接口實現類加載

SLF4J加載不同提供商的日志實現類

Spring

Spring中大量使用了SPI,比如:對servlet3.0規范對ServletContainerInitializer的實現、自動類型轉換Type Conversion SPI(Converter SPI、Formatter SPI)等

Dubbo

Dubbo中也大量使用SPI的方式實現框架的擴展, 不過它對Java提供的原生SPI做了封裝,允許用戶擴展實現Filter接口

3 使用介紹

要使用Java SPI,需要遵循如下約定:

1、當服務提供者提供了接口的一種具體實現后,在jar包的META-INF/services目錄下創建一個以“接口全限定名”為命名的文件,內容為實現類的全限定名;2、接口實現類所在的jar包放在主程序的classpath中;3、主程序通過java.util.ServiceLoder動態裝載實現模塊,它通過掃描META-INF/services目錄下的配置文件找到實現類的全限定名,把類加載到JVM;4、SPI的實現類必須攜帶一個不帶參數的構造方法;

示例代碼

步驟1、定義一組接口 (假設是org.foo.demo.IShout),并寫出接口的一個或多個實現,(假設是org.foo.demo.animal.Dog、org.foo.demo.animal.Cat)。

public interface IShout { void shout();}public class Cat implements IShout { @Override public void shout() { System.out.println('miao miao'); }}public class Dog implements IShout { @Override public void shout() { System.out.println('wang wang'); }}

步驟2、在 src/main/resources/ 下建立 /META-INF/services 目錄, 新增一個以接口命名的文件 (org.foo.demo.IShout文件),內容是要應用的實現類(這里是org.foo.demo.animal.Dog和org.foo.demo.animal.Cat,每行一個類)。

文件位置

- src -main -resources - META-INF- services - org.foo.demo.IShout

文件內容

org.foo.demo.animal.Dogorg.foo.demo.animal.Cat

步驟3、使用 ServiceLoader 來加載配置文件中指定的實現。

public class SPIMain { public static void main(String[] args) { ServiceLoader<IShout> shouts = ServiceLoader.load(IShout.class); for (IShout s : shouts) { s.shout(); } }}

代碼輸出:

wang wangmiao miao4 原理解析

首先看ServiceLoader類的簽名類的成員變量:

public final class ServiceLoader<S> implements Iterable<S>{private static final String PREFIX = 'META-INF/services/'; // 代表被加載的類或者接口 private final Class<S> service; // 用于定位,加載和實例化providers的類加載器 private final ClassLoader loader; // 創建ServiceLoader時采用的訪問控制上下文 private final AccessControlContext acc; // 緩存providers,按實例化的順序排列 private LinkedHashMap<String,S> providers = new LinkedHashMap<>(); // 懶查找迭代器 private LazyIterator lookupIterator; ......}

參考具體ServiceLoader具體源碼,代碼量不多,加上注釋一共587行,梳理了一下,實現的流程如下:

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

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

2 應用程序通過迭代器接口獲取對象實例ServiceLoader先判斷成員變量providers對象中(LinkedHashMap<String,S>類型)是否有緩存實例對象,如果有緩存,直接返回。如果沒有緩存,執行類的裝載,實現如下:(1) 讀取META-INF/services/下的配置文件,獲得所有能被實例化的類的名稱,值得注意的是,ServiceLoader可以跨越jar包獲取META-INF下的配置文件,具體加載配置的實現代碼如下:

try { String fullName = PREFIX + service.getName(); if (loader == null)configs = ClassLoader.getSystemResources(fullName); elseconfigs = loader.getResources(fullName); } catch (IOException x) { fail(service, 'Error locating configuration files', x); }

(2) 通過反射方法Class.forName()加載類對象,并用instance()方法將類實例化。(3) 把實例化后的類緩存到providers對象中,(LinkedHashMap<String,S>類型)然后返回實例對象。

5 總結

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

缺點:

雖然ServiceLoader也算是使用的延遲加載,但是基本只能通過遍歷全部獲取,也就是接口的實現類全部加載并實例化一遍。如果你并不想用某些實現類,它也被加載并實例化了,這就造成了浪費。獲取某個實現類的方式不夠靈活,只能通過Iterator形式獲取,不能根據某個參數來獲取對應的實現類。多個并發多線程使用ServiceLoader類的實例是不安全的。

以上就是深入理解Java中的SPI機制的詳細內容,更多關于Java SPI機制的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 污水提升器,污水提升泵,地下室排水,增压泵,雨水泵,智能供排水控制器-上海智流泵业有限公司 | SOUNDWELL 编码器|电位器|旋转编码器|可调电位器|编码开关厂家-广东升威电子制品有限公司 | 冷却塔降噪隔音_冷却塔噪声治理_冷却塔噪音处理厂家-广东康明冷却塔降噪厂家 | 万家财经_财经新闻_在线财经资讯网 | 私人别墅家庭影院系统_家庭影院音响_家庭影院装修设计公司-邦牛影音 | 共享雨伞_共享童车_共享轮椅_共享陪护床-共享产品的领先者_有伞科技 | 台湾Apex减速机_APEX行星减速机_台湾精锐减速机厂家代理【现货】-杭州摩森机电 | 贴板式电磁阀-不锈钢-气动上展式放料阀-上海弗雷西阀门有限公司 工业机械三维动画制作 环保设备原理三维演示动画 自动化装配产线三维动画制作公司-南京燃动数字 | 迪威娱乐|迪威娱乐客服|18183620002 | 东莞动力锂电池保护板_BMS智能软件保护板_锂电池主动均衡保护板-东莞市倡芯电子科技有限公司 | 压滤机-洗沙泥浆处理-压泥机-山东创新华一环境工程有限公司 | 吉祥新世纪铝塑板_生产铝塑板厂家_铝塑板生产厂家_临沂市兴达铝塑装饰材料有限公司 | 华禹护栏|锌钢护栏_阳台护栏_护栏厂家-华禹专注阳台护栏、楼梯栏杆、百叶窗、空调架、基坑护栏、道路护栏等锌钢护栏产品的生产销售。 | 超声波成孔成槽质量检测仪-压浆机-桥梁预应力智能张拉设备-上海硕冠检测设备有限公司 | 光泽度计_测量显微镜_苏州压力仪_苏州扭力板手维修-苏州日升精密仪器有限公司 | 细石混凝土泵_厂家_价格-烟台九达机械有限公司 | 天一线缆邯郸有限公司_煤矿用电缆厂家_矿用光缆厂家_矿用控制电缆_矿用通信电缆-天一线缆邯郸有限公司 | 120kv/2mA直流高压发生器-60kv/2mA-30kva/50kv工频耐压试验装置-旭明电工 | 西安微信朋友圈广告投放_微信朋友圈推广_西安度娘网络科技有限公司 | 塑料熔指仪-塑料熔融指数仪-熔体流动速率试验机-广东宏拓仪器科技有限公司 | 压装机-卧式轴承轮轴数控伺服压装机厂家[铭泽机械] | 作文导航网_作文之家_满分作文_优秀作文_作文大全_作文素材_最新作文分享发布平台 | 中式装修设计_全屋定制家具_实木仿古门窗花格厂家-喜迎门 | 杭州顺源过滤机械有限公司官网-压滤机_板框压滤机_厢式隔膜压滤机厂家 | 新材料分散-高速均质搅拌机-超声波分散混合-上海化烁智能设备有限公司 | 直齿驱动-新型回转驱动和回转支承解决方案提供商-不二传动 | 数字展示在线_数字展示行业门户网站| 博莱特空压机|博莱特-阿特拉斯独资空压机品牌核心代理商 | 半自动预灌装机,卡式瓶灌装机,注射器灌装机,给药器灌装机,大输液灌装机,西林瓶灌装机-长沙一星制药机械有限公司 | 依维柯自动挡房车,自行式国产改装房车,小型房车价格,中国十大房车品牌_南京拓锐斯特房车 - 南京拓锐斯特房车 | 铝箔袋,铝箔袋厂家,东莞铝箔袋,防静电铝箔袋,防静电屏蔽袋,防静电真空袋,真空袋-东莞铭晋让您的产品与众不同 | 免费分销系统 — 分销商城系统_分销小程序开发 -【微商来】 | 金联宇电缆总代理-金联宇集团-广东金联宇电缆实业有限公司 | 冷凝水循环试验箱-冷凝水试验箱-可编程高低温试验箱厂家-上海巨为(www.juweigroup.com) | 北京模型公司-军事模型-工业模型制作-北京百艺模型沙盘公司 | 水质监测站_水质在线分析仪_水质自动监测系统_多参数水质在线监测仪_水质传感器-山东万象环境科技有限公司 | 德国进口电锅炉_商用电热水器_壁挂炉_电采暖器_电热锅炉[德国宝] | 氢氧化钙设备_厂家-淄博工贸有限公司 | 齿轮减速马达一体式_蜗轮蜗杆减速机配电机-德国BOSERL齿轮减速电动机生产厂家 | 深圳富泰鑫五金_五金冲压件加工_五金配件加工_精密零件加工厂 | UV固化机_UVLED光固化机_UV干燥机生产厂家-上海冠顶公司专业生产UV固化机设备 |