Dubbo SPI機(jī)制分析【二】


title: Dubbo SPI機(jī)制分析【二】
tags: Dubbo,SPI,源碼
grammar_cjkRuby: true


上回Dubbo SPI機(jī)制分析【一】分析了ExtensionLoader.getExtensionLoader.getAdaptiveExtension的過(guò)程,接下來(lái)繼續(xù)分析ExtensionLoader.getExtension:

源碼分析

getExtension

    public T getExtension(String name) {
        if (name == null || name.length() == 0)
            throw new IllegalArgumentException("Extension name == null");
        if ("true".equals(name)) {
            return getDefaultExtension();
        }
        Holder<Object> holder = cachedInstances.get(name);
        if (holder == null) {
            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) {
                    //實(shí)例化拓展點(diǎn)實(shí)現(xiàn)類
                    instance = createExtension(name);
                    holder.set(instance);
                }
            }
        }
        return (T) instance;
    }

判斷當(dāng)前名字對(duì)應(yīng)的拓展點(diǎn)實(shí)現(xiàn)類的實(shí)例是否存在,如果不存在,進(jìn)行注冊(cè)式單例構(gòu)造實(shí)例createExtension

createExtension

    private T createExtension(String name) {
        //加載所有拓展點(diǎn)實(shí)現(xiàn)類妒峦,然后根據(jù)name獲取對(duì)應(yīng)的拓展點(diǎn)實(shí)現(xiàn)類
        Class<?> clazz = getExtensionClasses().get(name);
        if (clazz == null) {
            throw findException(name);
        }
        try {
            T instance = (T) EXTENSION_INSTANCES.get(clazz);
            if (instance == null) {
                //以注冊(cè)式單例 構(gòu)造實(shí)例
                EXTENSION_INSTANCES.putIfAbsent(clazz, clazz.newInstance());
                instance = (T) EXTENSION_INSTANCES.get(clazz);
            }
            //對(duì)實(shí)例進(jìn)行注入操作
            injectExtension(instance);
            //進(jìn)行包裝類封裝
            Set<Class<?>> wrapperClasses = cachedWrapperClasses;
            //遍歷當(dāng)前拓展點(diǎn)的所有包裝類,進(jìn)行包裝
            if (wrapperClasses != null && !wrapperClasses.isEmpty()) {
                for (Class<?> wrapperClass : wrapperClasses) {
                    instance = injectExtension((T) wrapperClass.getConstructor(type).newInstance(instance));
                }
            }
            return instance;
        } catch (Throwable t) {
            throw new IllegalStateException("Extension instance(name: " + name + ", class: " +
                    type + ")  could not be instantiated: " + t.getMessage(), t);
        }
    }

這段代碼一共做了這么幾件事:

  • 加載拓展點(diǎn)實(shí)現(xiàn)類

  • 實(shí)例化拓展點(diǎn)實(shí)現(xiàn)類麻车,并且進(jìn)行注入

  • 遍歷當(dāng)前拓展點(diǎn)的包裝類伯铣,并且將當(dāng)前name對(duì)應(yīng)的拓展點(diǎn)實(shí)例作為參數(shù)傳入該包裝類實(shí)例的構(gòu)造函數(shù)崎坊,將該包裝類實(shí)例化,如圖

    1539354624379.png

  • 最后返回實(shí)例

總結(jié)

至此壁公,getExtension分析完畢感论,主要需要注意的一點(diǎn)就是,通過(guò)此方法返回的實(shí)例最后可能會(huì)通過(guò)包裝類進(jìn)行包裝紊册,做一些特殊處理比肄,形成一個(gè)鏈?zhǔn)秸{(diào)用,最后才會(huì)真正調(diào)用到特定實(shí)例的具體方法囊陡。
那么還有個(gè)問(wèn)題芳绩,對(duì)于

ExtensionLoader.getExtensionLoader(Protocol.class).getAdaptiveExtension()

這段代碼,最后返回的Protocol類型撞反,具體是怎樣的呢妥色?
-首先通過(guò)getAdaptiveExtension獲取到的是Protocol$Adaptive:

package com.wl.dubbo;

/**
 * @Author: liumenglong
 * @Date: 2018/9/29 22:48
 * @Description:
 */

import com.alibaba.dubbo.common.extension.ExtensionLoader;

public class Protocol$Adaptive implements com.alibaba.dubbo.rpc.Protocol {
    public void destroy() {
        throw new UnsupportedOperationException("method public abstract void com.alibaba.dubbo.rpc.Protocol.destroy() of interface com.alibaba.dubbo.rpc.Protocol is not adaptive method!");
    }

    public int getDefaultPort() {
        throw new UnsupportedOperationException("method public abstract int com.alibaba.dubbo.rpc.Protocol.getDefaultPort() of interface com.alibaba.dubbo.rpc.Protocol is not adaptive method!");
    }

    public com.alibaba.dubbo.rpc.Invoker refer(java.lang.Class arg0, com.alibaba.dubbo.common.URL arg1) throws com.alibaba.dubbo.rpc.RpcException {
        if (arg1 == null) throw new IllegalArgumentException("url == null");
        com.alibaba.dubbo.common.URL url = arg1;
        //根據(jù)URL獲取到對(duì)應(yīng)的拓展名,如果沒(méi)有指定遏片,則默認(rèn)取“dubbo”
        String extName = (url.getProtocol() == null ? "dubbo" : url.getProtocol());
        if (extName == null)
            throw new IllegalStateException("Fail to get extension(com.alibaba.dubbo.rpc.Protocol) name from url(" + url.toString() + ") use keys([protocol])");
        //根據(jù)拓展名獲取對(duì)應(yīng)的拓展點(diǎn)實(shí)現(xiàn)類
        com.alibaba.dubbo.rpc.Protocol extension = (com.alibaba.dubbo.rpc.Protocol) ExtensionLoader.getExtensionLoader(com.alibaba.dubbo.rpc.Protocol.class).getExtension(extName);
        //調(diào)用實(shí)際拓展點(diǎn)的refer方法嘹害,如DubboProtocol
        return extension.refer(arg0, arg1);
    }

    public com.alibaba.dubbo.rpc.Exporter export(com.alibaba.dubbo.rpc.Invoker arg0) throws com.alibaba.dubbo.rpc.RpcException {
        if (arg0 == null) throw new IllegalArgumentException("com.alibaba.dubbo.rpc.Invoker argument == null");
        if (arg0.getUrl() == null)
            throw new IllegalArgumentException("com.alibaba.dubbo.rpc.Invoker argument getUrl() == null");
        com.alibaba.dubbo.common.URL url = arg0.getUrl();
        //根據(jù)URL獲取到對(duì)應(yīng)的拓展名,如果沒(méi)有指定丁稀,則默認(rèn)取“dubbo”
        String extName = (url.getProtocol() == null ? "dubbo" : url.getProtocol());
        if (extName == null)
            throw new IllegalStateException("Fail to get extension(com.alibaba.dubbo.rpc.Protocol) name from url(" + url.toString() + ") use keys([protocol])");
        //根據(jù)拓展名獲取對(duì)應(yīng)的拓展點(diǎn)實(shí)現(xiàn)類
        com.alibaba.dubbo.rpc.Protocol extension = 
                (com.alibaba.dubbo.rpc.Protocol) ExtensionLoader.
                        getExtensionLoader(com.alibaba.dubbo.rpc.Protocol.class).getExtension(extName);
        //調(diào)用實(shí)際拓展點(diǎn)的export方法吼拥,如DubboProtocol
        return extension.export(arg0);
    }
}

查看方法export,最后的調(diào)用還是需要先獲得具體的Protocol:

  //根據(jù)拓展名獲取對(duì)應(yīng)的拓展點(diǎn)實(shí)現(xiàn)類
        com.alibaba.dubbo.rpc.Protocol extension = 
                (com.alibaba.dubbo.rpc.Protocol) ExtensionLoader.
                        getExtensionLoader(com.alibaba.dubbo.rpc.Protocol.class).getExtension(extName);

那么也就是說(shuō)最后需要通過(guò)getExtension來(lái)獲取特定的Protocol,而此時(shí)extNamedubbo倚聚,那么應(yīng)該返回的就是DubboProtocol,然而在上面分析我們可以知道线衫,DubboProtcol在實(shí)例化之后,需要經(jīng)過(guò)包裝類的封裝惑折,因此通過(guò)此方法最后返回的ProtocolProtocolListenerWrapper

1539354624379.png

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末授账,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子惨驶,更是在濱河造成了極大的恐慌白热,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,651評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件粗卜,死亡現(xiàn)場(chǎng)離奇詭異屋确,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)续扔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門攻臀,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人纱昧,你說(shuō)我怎么就攤上這事刨啸。” “怎么了识脆?”我有些...
    開封第一講書人閱讀 162,931評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵设联,是天一觀的道長(zhǎng)善已。 經(jīng)常有香客問(wèn)我,道長(zhǎng)离例,這世上最難降的妖魔是什么换团? 我笑而不...
    開封第一講書人閱讀 58,218評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮宫蛆,結(jié)果婚禮上啥寇,老公的妹妹穿的比我還像新娘。我一直安慰自己洒扎,他們只是感情好辑甜,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評(píng)論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著袍冷,像睡著了一般磷醋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上胡诗,一...
    開封第一講書人閱讀 51,198評(píng)論 1 299
  • 那天邓线,我揣著相機(jī)與錄音,去河邊找鬼煌恢。 笑死骇陈,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的瑰抵。 我是一名探鬼主播你雌,決...
    沈念sama閱讀 40,084評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼二汛!你這毒婦竟也來(lái)了婿崭?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,926評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤肴颊,失蹤者是張志新(化名)和其女友劉穎氓栈,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體婿着,經(jīng)...
    沈念sama閱讀 45,341評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡授瘦,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了竟宋。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片提完。...
    茶點(diǎn)故事閱讀 39,731評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖袜硫,靈堂內(nèi)的尸體忽然破棺而出氯葬,到底是詐尸還是另有隱情,我是刑警寧澤婉陷,帶...
    沈念sama閱讀 35,430評(píng)論 5 343
  • 正文 年R本政府宣布帚称,位于F島的核電站官研,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏闯睹。R本人自食惡果不足惜戏羽,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評(píng)論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望楼吃。 院中可真熱鬧始花,春花似錦、人聲如沸孩锡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,676評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)躬窜。三九已至浇垦,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間荣挨,已是汗流浹背男韧。 一陣腳步聲響...
    開封第一講書人閱讀 32,829評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留默垄,地道東北人此虑。 一個(gè)月前我還...
    沈念sama閱讀 47,743評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像口锭,于是被迫代替她去往敵國(guó)和親朦前。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容