成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Dubbo中SPI機(jī)制的實(shí)現(xiàn)原理和優(yōu)勢(shì)

開發(fā) 前端
我們從 Dubbo 配置項(xiàng)的定義中發(fā)現(xiàn),Dubbo 采用了與 JDK 不同的實(shí)現(xiàn)機(jī)制。雖然 Dubbo 也采用了 SPI 機(jī)制,也是從 jar 包中動(dòng)態(tài)加載實(shí)現(xiàn)類,但它的實(shí)現(xiàn)方式與 JDK 中基于 ServiceLoader 是不一樣的。

確保系統(tǒng)的擴(kuò)展性是我們開展架構(gòu)設(shè)計(jì)工作的核心目標(biāo)之一。實(shí)現(xiàn)擴(kuò)展性的方法有很多,JDK 本身內(nèi)置了一個(gè) SPI(Service Provider Interface,服務(wù)提供者接口)機(jī)制,來幫開發(fā)人員動(dòng)態(tài)加載各種不同的實(shí)現(xiàn)類,只要這些實(shí)現(xiàn)類遵循一定的開發(fā)規(guī)范即可。

另一方面,JDK 自帶的 SPI 機(jī)制存在一定的缺陷,因此市面上有些框架對(duì) JDK 中的 SPI 機(jī)制做了一些增強(qiáng),這方面的代表性框架就是 Dubbo。在今天的課程中,我們將對(duì)這兩種 SPI 機(jī)制進(jìn)行對(duì)比,并重點(diǎn)闡述 Dubbo 中 SPI 機(jī)制的實(shí)現(xiàn)原理和優(yōu)勢(shì)。為了更好地做比較,讓我們先從 JDK 中的 SPI 機(jī)制講起。

JDK 中的 SPI 機(jī)制解析

如果我們采用 JDK 中的 SPI,具體的開發(fā)工作會(huì)涉及三個(gè)步驟。

實(shí)現(xiàn) JDK SPI 機(jī)制的開發(fā)步驟實(shí)現(xiàn) JDK SPI 機(jī)制的開發(fā)步驟

對(duì)于 SPI 的開發(fā)者而言,我們需要設(shè)計(jì)一個(gè)服務(wù)接口,然后根據(jù)業(yè)務(wù)場(chǎng)景提供不同的實(shí)現(xiàn)類,這是第一步。

接下來的第二步是關(guān)鍵,我們需要?jiǎng)?chuàng)建一個(gè)以服務(wù)接口命名的配置文件,并把這個(gè)文件放置到代碼工程的 META-INF/services 目錄下。請(qǐng)注意,在這個(gè)配置文件中,我們需要指定服務(wù)接口對(duì)應(yīng)實(shí)現(xiàn)類的完整類名。通過這一步,我們可以得到了一個(gè)包含 SPI 類和配置的 jar 包。

最后,SPI 的使用者就可以加載這個(gè) jar 包并找到其中的這個(gè)配置文件,并根據(jù)所配置的實(shí)現(xiàn)類完整類名對(duì)這些類進(jìn)行實(shí)例化。

上圖中的后面兩個(gè)步驟實(shí)際上都是為了遵循 JDK 中 SPI 的實(shí)現(xiàn)機(jī)制而進(jìn)行的配置工作。

為了實(shí)現(xiàn)對(duì) SPI 實(shí)現(xiàn)類的動(dòng)態(tài)記載,JDK 專門提供了一個(gè) ServiceLoader 工具類,這個(gè)工具類的使用方法如下所示:

public static void main(String[] args) {
  ServiceLoader<LogProvider> loader = ServiceLoader.load(LogProvider.class);
  for (LogProvider provider : loader) {
   System.out.println(provider.getClass());
   provider.info(“testInfo”);
  }
}

這里有一個(gè) LogProvider 接口,并通過 ServiceLoader 的 load 方法將這個(gè)接口所配置的實(shí)現(xiàn)類加載到內(nèi)存中,從而可以方便地使用這些 SPI 實(shí)現(xiàn)類所提供的功能。

接下來,讓我們來分析一下這個(gè) ServiceLoader 工具類的實(shí)現(xiàn)原理。

ServiceLoader 本身實(shí)現(xiàn)了 JDK 中的 Iterable 接口,因此在上面的代碼示例中,通過 ServiceLoader.load 方法我們獲取的是一個(gè)迭代器,而底層則用到了 ServiceLoader.LazyIterator 這個(gè)迭代器類。

從命名上看,LazyIterator 是一個(gè)具備延遲加載機(jī)制的迭代器,它有 hasNextService 和 nexServicet 這兩個(gè)核心方法。我們先來看 hasNextService 方法:

//配置文件路徑
static final String PREFIX = "META-INF/services/";
private boolean hasNextService() {
    if (nextName != null) {
        return true;
    }
    if (configs == null) {
        // 通過 PREFIX 前綴與服務(wù)接口的名稱,我們可以找到目標(biāo) SPI 配置文件
        String fullName = PREFIX + service.getName();
        // 加載配置文件
        if (loader == null)
            configs = ClassLoader.getSystemResources(fullName);
        else
            configs = loader.getResources(fullName);
    }
    // 對(duì) SPI 配置文件進(jìn)行遍歷,并解析配置內(nèi)容
    while ((pending == null) || !pending.hasNext()) {
        if (!configs.hasMoreElements()) {
            return false;
        }
        // 解析配置文件
        pending = parse(service, configs.nextElement());
 }
 // 更新 nextName 字段
    nextName = pending.next();
    return true;
}

可以看到,hasNextService 方法的核心作用是找到并解析配置文件。而接下來要展開的 nextService 方法則負(fù)責(zé)對(duì)所配置的類進(jìn)行實(shí)例化,核心實(shí)現(xiàn)如下所示:

private S nextService() {
    String cn = nextName;
    nextName = null;
    // 加載 nextName 字段指定的類
 Class<?> c = Class.forName(cn, false, loader);
 // 檢測(cè)類型
    if (!service.isAssignableFrom(c)) {
        fail(service, "Provider " + cn  + " not a subtype");
 }
 // 創(chuàng)建實(shí)現(xiàn)類的對(duì)象
 S p = service.cast(c.newInstance());
 // 緩存已創(chuàng)建的對(duì)象
 providers.put(cn, p);
    return p;
}

這里通過 newInstance 方法創(chuàng)建了目標(biāo)實(shí)例,并將已創(chuàng)建的實(shí)例對(duì)象放到 providers 集合中進(jìn)行緩存,從而提高訪問效率。

Dubbo 中的 SPI 機(jī)制解析

為了實(shí)現(xiàn)框架自身的擴(kuò)展性,Dubbo 也采用了類似 JDK 中 SPI 的設(shè)計(jì)思想,但提供了一套新的實(shí)現(xiàn)方式,并添加了一些擴(kuò)展功能。

Dubbo 中與 SPI 機(jī)制相關(guān)的注解主要包括@SPI、@Adaptive 和@Activate,其中@SPI 注解提供了與 JDK 中 SPI 類似的功能。

Dubbo 中 SPI 相關(guān)注解Dubbo 中 SPI 相關(guān)注解

這三個(gè)注解的應(yīng)用場(chǎng)景各不相同,其中@SPI 注解為 Dubbo 提供了最基礎(chǔ)的 SPI 機(jī)制,而@Adaptive 和@Activate 注解都是構(gòu)建在這個(gè)注解之上,因此我們重點(diǎn)介紹@SPI 注解。如果在某個(gè)接口上添加了這個(gè)注解,那么 Dubbo 在運(yùn)行過程中就會(huì)去查找接口對(duì)應(yīng)的擴(kuò)展點(diǎn)實(shí)現(xiàn)。

在 Dubbo 中,隨處可以看到@SPI 注解的應(yīng)用場(chǎng)景。舉個(gè)例子,Protocol 接口定義如下:

@SPI("dubbo")
public interface Protocol

可以看到,在這個(gè)接口上使用的就是@SPI(“dubbo”) 注解。

請(qǐng)注意,在@SPI 注解中可以指定默認(rèn)擴(kuò)展點(diǎn)的名稱,例如這里的“dubbo”用來表明在 Protocol 接口的所有實(shí)現(xiàn)類中,DubboProtocol 是它的默認(rèn)實(shí)現(xiàn)。

有了 SPI 的定義,我們接下來看一看 Dubbo 中 SPI 配置信息的存儲(chǔ)方式。我們已經(jīng)知道,JDK 只會(huì)把 SPI 配置存放在 META-INF/services/這個(gè)目錄下,而 Dubbo 則提供了三個(gè)類似這樣的目錄:

Dubbo 中 SPI 配置的存放目錄Dubbo 中 SPI 配置的存放目錄

作為示例,我們繼續(xù)圍繞上面提到的 Protocol 接口展開討論。

針對(duì) Protocol 接口,Dubbo 提供了 gRPCProtocol、DubboProtocol 等多個(gè)實(shí)現(xiàn)類,并通過 SPI 機(jī)制完成對(duì)具體某種實(shí)現(xiàn)方案的加載過程。讓我們分別來到提供這些實(shí)現(xiàn)類的代碼工程 dubbo-rpc-grpc 和 dubbo-rpc-dubbo,會(huì)發(fā)現(xiàn)在 META-INF/dubbo/internal/目錄下都包含了一個(gè) com.apache.dubbo.rpc.Protocol 配置文件。其中,dubbo-rpc-grpc 工程的代碼結(jié)構(gòu)如圖所示:

dubbo-rpc-grpc 工程的代碼結(jié)構(gòu)dubbo-rpc-grpc 工程的代碼結(jié)構(gòu)

類似的,dubbo-rpc-dubbo 工程的代碼結(jié)構(gòu)如下圖所示:

dubbo-rpc-dubbo 工程的代碼結(jié)構(gòu)dubbo-rpc-dubbo 工程的代碼結(jié)構(gòu)

我們分別打開這兩個(gè)工程的 com.apache.dubbo.rpc.Protocol 配置文件,可以發(fā)現(xiàn)它們分別指向了 org.apache.dubbo.rpc.protocol.grpc.GrpcProtocol 和 org.apache.dubbo.rpc.protocol.dubbo.DubboProtocol 類。

//dubbo-rpc-grpc 工程
grpc=org.apache.dubbo.rpc.protocol.grpc.GrpcProtocol
//dubbo-rpc-dubbo 工程:
dubbo=org.apache.dubbo.rpc.protocol.dubbo.DubboProtocol

當(dāng) Dubbo 在引用具體某一個(gè)代碼工程時(shí),就可以通過這個(gè)工程中的配置項(xiàng)就可以找到 Dubbo 接口對(duì)應(yīng)的擴(kuò)展點(diǎn)實(shí)現(xiàn)。

同時(shí),我們從上面配置項(xiàng)中也可以看出,Dubbo 中采用的定義方式與 JDK 中的不一樣。Dubbo 使用的一個(gè) Key 值(如上面的 gRPC 和 Dubbo)來指定具體的配置項(xiàng)名稱,而不是采用完整類路徑。

介紹完@SPI 注解,我們接下來看 Dubbo 中的 ExtensionLoader 類,這個(gè)類扮演著與 JDK 中 ServiceLoader 工具類相同的角色。ExtensionLoader 是實(shí)現(xiàn)擴(kuò)展點(diǎn)加載的核心類,如果我們想要獲取 DubboProtocol 這個(gè)實(shí)現(xiàn)類,那么可以采用以下方式:

DubboProtocol dubboProtocol = ExtensionLoader.getExtensionLoader(Protocol.class).getExtension(DubboProtocol.NAME);

我們來看一下這里 getExtension 方法的細(xì)節(jié),這個(gè)方法代碼如下所示:

public T getExtension(String name) {
     ...
     //從緩存中獲取目標(biāo)對(duì)象
        Holder<Object> holder = cachedInstances.get(name);
        if (holder == null) {
         //將目標(biāo)對(duì)象放到緩存中
            cachedInstances.putIfAbsent(name, new Holder<Object>());
            holder = cachedInstances.get(name);
        }
        Object instance = holder.get();
        if (instance == null) {
            synchronized (holder) {
                instance = holder.get();
                if (instance == null) {
                 //創(chuàng)建目標(biāo)對(duì)象
                    instance = createExtension(name);
                    holder.set(instance);
                }
            }
        }
        return (T) instance;
}

我們看到這里同樣用到了緩存機(jī)制。這個(gè)方法會(huì)首先檢查緩存中是否已經(jīng)存在擴(kuò)展點(diǎn)實(shí)例,如果沒有則通過 createExtension 方法進(jìn)行創(chuàng)建。我們一路跟蹤 createExtension 方法,終于看到了熟悉的 SPI 機(jī)制,如下所示:

private Map<String, Class<?>> loadExtensionClasses() {
        final SPI defaultAnnotation = type.getAnnotation(SPI.class);
        if (defaultAnnotation != null) {
             //確定緩存名稱
        }
        Map<String, Class<?>> extensionClasses = new HashMap<String, Class<?>>();
     //分別從三個(gè)目錄中加載類實(shí)例
        loadFile(extensionClasses, DUBBO_INTERNAL_DIRECTORY);
        loadFile(extensionClasses, DUBBO_DIRECTORY);
        loadFile(extensionClasses, SERVICES_DIRECTORY);
        return extensionClasses;
}

在這里,我們調(diào)用了三次 loadFile 方法,分別在 META-INF/dubbo/、META-INF/services/和 META-INF/dubbo/internal/這三個(gè)目錄中加載擴(kuò)展點(diǎn)。在 loadFile 方法中,Dubbo 是直接通過 Class.forName 的形式加載這些 SPI 的擴(kuò)展類,并進(jìn)行緩存。

講到這里,我們發(fā)現(xiàn),為了提升實(shí)例類的加載速度,Dubbo 和 JDK 都采用了緩存機(jī)制,這是它們的一個(gè)共同點(diǎn)。但實(shí)際上,我們也已經(jīng)可以梳理 Dubbo 中 SPI 機(jī)制與 JDK 中 SPI 機(jī)制的區(qū)別,核心有兩點(diǎn),就是 配置文件位置和 獲取實(shí)現(xiàn)類的條件

Dubbo 與 JDK 中 SPI 機(jī)制的兩點(diǎn)核心區(qū)別Dubbo 與 JDK 中 SPI 機(jī)制的兩點(diǎn)核心區(qū)別

  • 從加載 SPI 實(shí)例的配置文件位置來看,Dubbo 支持更多的加載路徑。JDK 只能加載一個(gè)固定的 META-INF/services,而 Dubbo 有三個(gè)路徑。
  • 就獲取實(shí)現(xiàn)類的條件而言,Dubbo 采用的是直接通過名稱對(duì)應(yīng)的 Key 值來定位具體實(shí)現(xiàn)類,而 ServiceLoader 內(nèi)部使用的是一個(gè)迭代器,在獲取目標(biāo)接口的實(shí)現(xiàn)類時(shí),只能通過遍歷的方式把配置文件中的類全部加載并實(shí)例化,顯然這樣效率比較低下。

簡(jiǎn)單來說,Dubbo 沒有直接沿用 JDK SPI 機(jī)制,而是自己實(shí)現(xiàn)一套的主要目的就是克服這種效率低下的情況,并提供了更多的靈活性。

總結(jié)

我們從 Dubbo 配置項(xiàng)的定義中發(fā)現(xiàn),Dubbo 采用了與 JDK 不同的實(shí)現(xiàn)機(jī)制。雖然 Dubbo 也采用了 SPI 機(jī)制,也是從 jar 包中動(dòng)態(tài)加載實(shí)現(xiàn)類,但它的實(shí)現(xiàn)方式與 JDK 中基于 ServiceLoader 是不一樣的。于是,我們?cè)敿?xì)分析了 JDK 和 Dubbo 在 SPI 機(jī)制設(shè)計(jì)和實(shí)現(xiàn)上的差異,并闡明了 Dubbo 內(nèi)部的實(shí)現(xiàn)原理和所具備的優(yōu)勢(shì)。

責(zé)任編輯:武曉燕 來源: 程序員技術(shù)充電站
相關(guān)推薦

2024-10-29 08:34:55

SPI機(jī)制接口

2025-05-08 03:25:00

DubboSPI機(jī)制

2021-09-10 08:31:19

DubboSPI框架

2020-12-14 11:35:22

SPI Java機(jī)制

2023-12-11 07:21:12

SPI機(jī)制插件

2025-03-04 09:02:25

JavaSPI機(jī)制

2024-01-15 08:25:53

SPI機(jī)制JavaDubbo

2018-07-06 15:30:14

DubboSPIJDK

2023-03-13 22:09:59

JavaSpring機(jī)制

2020-06-30 15:35:36

JavaSPI代碼

2023-02-15 13:57:13

JavaSPI動(dòng)態(tài)擴(kuò)展

2011-11-30 14:35:19

JavaSPI

2025-05-27 01:00:00

2011-06-13 10:21:25

QT 信號(hào) 槽機(jī)制

2020-11-20 07:51:02

JavaSPI機(jī)制

2010-04-27 12:56:35

lvs負(fù)載均衡

2023-06-05 08:07:33

JavaJava SPI

2023-01-30 22:43:39

DubboZooKeeper

2021-08-28 09:06:11

Dubbo架構(gòu)服務(wù)

2009-08-28 10:18:48

Java序列化
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 国产免费看 | 日韩在线精品 | 99热在这里只有精品 | 中文在线一区二区 | 精品成人 | 偷拍亚洲色图 | 日韩精品在线免费观看 | 国产精品久久久久久久久久久久午夜片 | 一区二区三区高清 | 日韩1区| 国产欧美日韩综合精品一 | 天天干b | av片免费| 亚洲狠狠爱 | av三级在线观看 | 久久精点视频 | 99精品视频一区二区三区 | 毛片毛片毛片毛片毛片 | 在线视频国产一区 | 日韩精品久久久 | 午夜成人免费视频 | 国产免费一区二区 | 日韩av中文 | 成人在线视 | 久久精品网 | 四色成人av永久网址 | 在线 丝袜 欧美 日韩 制服 | 日韩二| 99福利视频导航 | 一区二区三区四区免费视频 | 天天天操操操 | 国产精品久久久久久一区二区三区 | 一区二区手机在线 | 精品一区二区三区四区 | 正在播放亚洲 | 欧美一区视频 | 久久av资源网 | 久久夜视频 | 国产精品成人品 | 一区免费 | 一区二区不卡视频 |