【Spring】Java SPI機制及Spring Boot使用實例

目錄

一、SPI是什么

1.1 SPI 和 API 有什么區別?

二、使用場景

三、使用介紹

四、Spring Boot實例運用

五、總結


一、SPI是什么

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

整體機制圖如下:

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

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

Java SPI就是提供這樣的一個機制:為某個接口尋找服務實現的機制。有點類似IoC的思想,就是將裝配的控制權移到程序之外,在模塊化設計中這個機制尤其重要。所以SPI的核心思想就是解耦

涉及到SPI的地方就是打破了雙親委派機制:2、線程上下文類加載器破壞雙親委派模型

1.1 SPI API 有什么區別?

那 SPI 和 API 有啥區別?

說到 SPI 就不得不說一下 API 了,從廣義上來說它們都屬于接口,而且很容易混淆。下面先用一張圖說明一下:

一般模塊之間都是通過通過接口進行通訊,那我們在服務調用方和服務實現方(也稱服務提供者)之間引入一個“接口”。

當實現方提供了接口和實現,我們可以通過調用實現方的接口從而擁有實現方給我們提供的能力,這就是 API ,這種接口和實現都是放在實現方的。

當接口存在于調用方這邊時,就是 SPI ,由接口調用方確定接口規則,然后由不同的廠商去根據這個規則對這個接口進行實現,從而提供服務

舉個通俗易懂的例子:公司 H 是一家科技公司,新設計了一款芯片,然后現在需要量產了,而市面上有好幾家芯片制造業公司,這個時候,只要 H 公司指定好了這芯片生產的標準(定義好了接口標準),那么這些合作的芯片公司(服務提供者)就按照標準交付自家特色的芯片(提供不同方案的實現,但是給出來的結果是一樣的)。

二、使用場景

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

比較常見的例子:

  • 數據庫驅動加載接口實現類的加載:JDBC加載不同類型數據庫的驅動
  • 日志門面接口實現類加載:SLF4J加載不同提供商的日志實現類
  • Spring:Spring中大量使用了SPI,比如:對servlet3.0規范對ServletContainerInitializer的實現、自動類型轉換Type Conversion SPI(Converter SPI、Formatter SPI)等
  • Dubbo:Dubbo中也大量使用SPI的方式實現框架的擴展, 不過它對Java提供的原生SPI做了封裝,允許用戶擴展實現Filter接口

三、使用介紹

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

  • 當服務提供者提供了接口的一種具體實現后,在jar包的META-INF/services目錄下創建一個以“接口全限定名”為命名的文件,內容為實現類的全限定名
  • 接口實現類所在的jar包放在主程序的classpath中;
  • 主程序通過java.util.ServiceLoder動態裝載實現模塊,它通過掃描META-INF/services目錄下的配置文件找到實現類的全限定名,把類加載到JVM
  • 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 {@Overridepublic void shout() {System.out.println("miao miao");}
}public class Dog implements IShout {@Overridepublic 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.Dog
org.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 wang
* miao miao
*/

四、Spring Boot實例運用

Spring Boot相信很多人都用過,在spring-boot和spring-boot-autoconfigure這兩個jar包的META-INF/spring.factories路徑下,保存的就是Spring Boot使用SPI機制配置的屬性,里面有Spring Boot運行時需要讀取的類,包括EnableAutoConfiguration等自動配置類,其部分關鍵配置如下:

# PropertySource Loaders
org.springframework.boot.env.PropertySourceLoader=\
org.springframework.boot.env.PropertiesPropertySourceLoader,\
org.springframework.boot.env.YamlPropertySourceLoader# Application Listeners
org.springframework.context.ApplicationListener=\
org.springframework.boot.ClearCachesApplicationListener,\
org.springframework.boot.builder.ParentContextCloserApplicationListener,\
org.springframework.boot.context.FileEncodingApplicationListener,\
org.springframework.boot.context.config.AnsiOutputApplicationListener,\
org.springframework.boot.context.config.ConfigFileApplicationListener,\

在這里面配置了PropertySourceLoader和ApplicationListener等接口的具體實現類,然后通過SpringFactoriesLoader這個類去加載這個文件,并獲得具體的類路徑。

SpringFactoriesLoader其部分關鍵源碼如下:

public final class SpringFactoriesLoader {// 加載器所需要加載的路徑public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {MultiValueMap<String, String> result = cache.get(classLoader);if (result != null) {return result;}try {// 根據路徑去錄取各個包下的文件Enumeration<URL> urls = (classLoader != null ?classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));result = new LinkedMultiValueMap<>();// 獲取后進行循環遍歷,因為不止一個包有spring.factories文件while (urls.hasMoreElements()) {URL url = urls.nextElement();UrlResource resource = new UrlResource(url);Properties properties = PropertiesLoaderUtils.loadProperties(resource);// 獲取到了key和value對應關系for (Map.Entry<?, ?> entry : properties.entrySet()) {String factoryClassName = ((String) entry.getKey()).trim();// 循環獲取配置文件的value,并放進result集合中for (String factoryName : StringUtils.commaDelimitedListToStringArray((String) entry.getValue())) {result.add(factoryClassName, factoryName.trim());}}}// 并緩存起來,以便后續直接獲取cache.put(classLoader, result);return result;}catch (IOException ex) {...}}
}

當開發者獲取到這些key-value后,便可以直接使用Class.forName()方法獲取Class對象,接著使用Class實例化便可以完成基于接口的編程+策略模式+配置文件這種搭配模式了。

Spring Boot依賴的很多包中都有spring.factories,用于告知Spring Boot需要自動配置的實現類有哪些:

五、總結

優點:

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

缺點:

  • 雖然ServiceLoader也算是使用的延遲加載,但是基本只能通過遍歷全部獲取,也就是接口的實現類全部加載并實例化一遍。如果你并不想用某些實現類,它也被加載并實例化了,這就造成了浪費(Spring Boot中進行了優化,會將要自動配置的類先去重,再過濾,只把真正要用的類進行自動配置)。獲取某個實現類的方式不夠靈活,只能通過Iterator形式獲取,不能根據某個參數來獲取對應的實現類。
  • 多個并發多線程使用ServiceLoader類的實例是不安全的。

相關文章:【JVM筆記】如何打破雙親委派機制?-CSDN博客

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/913891.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/913891.shtml
英文地址,請注明出處:http://en.pswp.cn/news/913891.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

多維度數據資產測繪技術在安全管控平臺中的應用實踐

一、數據資產治理困境&#xff1a;從 “黑箱” 到 “可見性” 的行業挑戰在數字化轉型加速的當下&#xff0c;企業數據資產呈現爆發式增長&#xff0c;而傳統資產梳理手段因維度單一、時效性差&#xff0c;導致 “資產黑箱” 問題頻發。某省級運營商曾在安全評估中發現&#xf…

搭建react18+項目過程中遇到的問題(vite)

問題1. 頁面中使用import.meta.env獲取環境變量有紅色波浪線提示錯誤按提示給ts.config.ts文件中的compilerOptions增加了"module": “esnext” (es2020 | es2022 | system)這幾個也不行 但是另一個問題出現了安裝的第三方庫引入報錯了 按照提示我們將module改成了’…

Linux epoll簡介與C++TCP服務器代碼示例

Linux epoll 簡介與示例 TCP 服務器 1. 為什么要用 epoll select/poll 每次調用都把全部文件描述符從用戶態拷貝到內核態,隨連接數增長而線性變慢;epoll 采用事件驅動+就緒隊列的方式,內核只把“已就緒”的描述符返回給用戶態,O(1) 規模擴展;支持 邊沿觸發 Edge-Triggere…

IPv4和IPv6雙棧配置

根據IPv6的學習&#xff0c;完成以下一個簡單的雙棧配置案例&#xff0c;具體結構如下圖所示。PC1的 IPv4&#xff1a;192.168.2.1/24 、IPv6&#xff1a;2001:db8:2::2/64&#xff0c;PC2的 IPv4&#xff1a;192.168.3.1/24 、IPv6&#xff1a;2001:db8:3::2/64總共需要兩臺PC…

Robyn高性能Web框架系列08:使用 Rust 擴展 Robyn

使用 Rust 擴展 RobynPyO3 Bridge示例&#xff1a;一個簡單的Rust擴展1、安裝必須的組件2、初始化Rust項目3、編寫Rust代碼4、在Robyn中使用Rust代碼在“Robyn高性能Web框架系列07&#xff1a;多進程、性能調優”一節中&#xff0c;我們講解了Robyn豐富的性能調優方式&#xff…

利用Pandas進行條件替換與向前填充

目錄一、需求二、實現代碼案例代碼詳細解釋1. 導入庫和創建數據2. 條件替換與填充a. 條件掩碼 - mask()b. 向前填充 - ffill()c. 類型轉換 - astype(int)3. 打印結果三、實際應用場景四、可能的變體五、總結一、需求 示例數據&#xff1a; 項 目 0 1 0 1 0 1 2 0 2 3 …

springboot數據脫敏(接口級別)

文章目錄自定義脫敏注解脫敏注解接口脫敏注解反射AOP實現字段脫敏切面定義脫敏策略脫敏策略的接口電話號碼脫敏策略郵箱脫敏不脫敏姓名脫敏身份證號脫敏JacksonAOP實現脫敏定義序列化序列化實現脫敏切面定義JacksonThreadLocal攔截器實現脫敏定義ThreadLocal自定義序列化序列化…

Spring核心原理的快速入門:快速了解IoC與DI

IoC IoC&#xff1a;Inversion of Control(控制反轉) Spring是一個包含了眾多工具的IoC容器(即bean&#xff1a;spring管理的對象),也就是說Spring 是一個“控制反轉”的容器。 之前是對象本身管理自己的生命周期等等&#xff0c;現在交給spring來管理對象的生命周期 IoC介紹 …

ffmpeg 中config 文件一些理解

依賴檢查 config中看到最多的是&#xff1a; ... nvenc_deps"ffnvcodec" nvenc_deps_any"libdl LoadLibrary" nvenc_encoder_deps"nvenc" ... h264_crystalhd_decoder_select"crystalhd h264_mp4toannexb_bsf h264_parser" h264_cuvid…

Digital Rainwater Collection System (v1.0)

The law doesn’t punish the masses. If only one guy runs his own rainwater system, he gets fined for “illegal mining.” But if millions of households self-host their “digital wells,” the whole centralized model collapses. Cloud providers and regulators …

NFS文件存儲及部署論壇(小白的“升級打怪”成長之路)

目錄 一、概述 NFS掛載原理 NFS工作原理 RPC與NFS通訊過程 二、NFS服務安裝與啟停 NFS服務安裝 NFS服務啟停 三、NFS服務配置文件 四、NFS文件共享配置文件 配置參數說明 五、命令解析 六、客戶端訪問 七、客戶端掛載 實戰案例 部署NFS文件存儲及discuz論壇應用 …

JavaScript 對象創建:new 操作符全解析

引言 在 JavaScript 中&#xff0c;new 操作符是實現面向對象編程的??核心機制??之一。本文將從原理層面對 new 操作符進行深度剖析&#xff0c;探討其工作機制、內部實現和實際應用場景。無論您是 JavaScript 初學者還是資深開發者&#xff0c;都能從本文獲得以下知識和技…

Spring Boot + Vue.js 全棧開發:從前后端分離到高效部署,打造你的MVP利器!

文章目錄一、為何選擇 Spring Boot Vue.js&#xff1f;全棧開發的“黃金搭檔”&#xff01;二、項目初始化與基礎架構搭建2.1 后端&#xff1a;初始化 Spring Boot 項目2.2 前端&#xff1a;初始化 Vue.js 項目2.3 核心配置&#xff1a;打通前后端通信與跨域&#xff01;后端 …

容器技術技術入門與Docker環境部署

目錄 一&#xff1a;Docker 概述 1&#xff1a;什么是Docker 2:Docker 的優勢 3&#xff1a;Docker的應用場景 4&#xff1a;Docker核心概念 二&#xff1a;Docker 安裝 三&#xff1a;Docker 鏡像操作 1&#xff1a;獲取鏡像 2&#xff1a;查看鏡像信息 3&#xff1a…

構建高效分布式系統:bRPC組合Channels與HTTP/H2訪問指南

構建高效分布式系統&#xff1a;bRPC組合Channels與HTTP/H2訪問指南 引言 在現代分布式系統中&#xff0c;下游服務訪問的復雜性日益增加。bRPC通過組合Channels和HTTP/H2訪問優化&#xff0c;提供了解決多層級RPC調用、負載均衡和協議兼容性問題的完整方案。本文將深入解析兩大…

WSL創建Ubuntu子系統與 VS code 開發

文章目錄一、打開Windows的虛擬化基礎功能二、安裝WSL和Ubuntu1. 安裝 WSL2. 安裝 Ubuntu三、 VScode一、打開Windows的虛擬化基礎功能 控制面板-程序和功能-啟動或關閉Windows功能&#xff0c;勾選適用于Linux的Windows子系統、虛擬機平臺&#xff0c; 完成后根據提示重啟電腦…

AlpineLinux二進制文件部署prometheus

在Alpine Linux上通過二進制文件部署Prometheus的步驟如下: 創建用戶和組: groupadd prometheus useradd -g prometheus -m -s /sbin/nologin prometheus下載Prometheus二進制文件: 你可以從Prometheus的官方GitHub發布頁面下載最新的二進制文件。例如,使用wget命令: wget…

IoT 小程序:如何破解設備互聯的碎片化困局?

一、IoT 設備管理為何需要輕量化解決方案&#xff1f;隨著物聯網設備規模爆發式增長 —— 預計 2025 年全球連接數將達 270 億臺&#xff0c;傳統 Native 應用開發模式的弊端日益凸顯&#xff1a;某智能家居廠商開發 3 款主流設備 APP&#xff0c;需維護 iOS/Android/ 小程序 3…

Word 怎么讓字變大、變粗、換顏色?

這是Word中最常用也最基礎的操作之一。學會它&#xff0c;你的文檔就會立刻變得重點突出&#xff0c;清晰易讀。 記住一個核心前提&#xff1a;無論做什么格式修改&#xff0c;第一步永遠是【先選中你要修改的文字】。 你可以把鼠標放在文字的開頭&#xff0c;按住左鍵&#xf…

Ruby 安裝 - Linux

Ruby 安裝 - Linux 引言 Ruby 是一種廣泛使用的高級編程語言,以其簡潔、優雅和強大的功能而聞名。在 Linux 系統上安裝 Ruby 是許多開發者的首要任務。本文將詳細介紹如何在 Linux 系統上安裝 Ruby,包括準備工作、安裝過程和常見問題解決。 準備工作 在開始安裝 Ruby 之前…