Dubbo源碼解析 — 服務引用原理

前言

經過上一篇dubbo源碼解析-簡單原理、與spring融合的鋪墊,我們已經能簡單的實現了dubbo的服務引用.其實上一篇中的代碼,很多都是從dubbo源碼中復制出來,甚至有些類名,變量名都沒改.那請問,我為什么要這么做?

我認為學習一個框架,無非就三個步驟.

掌握基本使用

看過源碼,知道其中原理

臨摹源碼,自己仿寫一個簡易的框架

其實大家都清楚,編程這東西,最關鍵是多動手.也就是,第三步才是最關鍵的.但是現實也是非常殘酷的,絕大多數人都停留在第一步.光是第二步,都有些讓人產生的心里恐懼.所以在寫服務引用的時候,我就想到了小時候看紀曉嵐的一個片段.當時紅樓夢是禁書,紀曉嵐為了讓太后看紅樓夢,就把紅樓夢這個名字換成了石頭記.這樣太后自然就沒有心里負擔.我覺得用一個圖來描述可能更貼切

當然臨摹源碼的這個過程,依肥朝拙見,也需要分為三個過程,分別是入門版(用最簡單的代碼表達出框架原理)、進階版(加入設計模式等思想,在入門版的基礎上優(yōu)化代碼)、高級版(和框架代碼基本一致).

當然上一篇的入門版只是拋磚引玉,等整個dubbo源碼解析系列完結之后,和大家一起臨摹dubbo源碼也在計劃當中.當然更多后續(xù)進展關注肥朝即可.

插播面試題

描述一下dubbo服務引用的過程,原理

既然你提到了dubbo的服務引用中封裝通信細節(jié)是用到了動態(tài)代理,那請問創(chuàng)建動態(tài)代理常用的方式有哪些,他們又有什么區(qū)別?dubbo中用的是哪一種?(高頻題)

除了JDK動態(tài)代理和CGLIB動態(tài)代理外,還知不知道其他實現代理的方式?(區(qū)分度高)

原諒他

看源碼對于大多數人來說,最難的一點莫過于"從源碼的哪個地方開始看".雖然我之前數十篇dubbo源碼解析都在回答這個問題,但是每發(fā)出一篇,都還是有小伙伴私信問我同樣的問題.對此,我當然是選擇"原諒他".因此,本篇我又再次粗暴式的點題,"怎么看源碼".就把本篇來說,這個服務引用的原理,我們要從哪里開始看呢?我們一起看一下官方文檔

如果你在上一篇中把我貼出來的demo都實現過一遍,再看到這個圖,就不難總結出服務引用無非就是做了兩件事

將spring的schemas標簽信息轉換bean,然后通過這個bean的信息,連接、訂閱zookeeper節(jié)點信息創(chuàng)建一個invoker

將invoker的信息創(chuàng)建一個動態(tài)代理對象

溫馨提示:除了看官方文檔入手,在dubbo源碼解析-服務暴露原理中我還提到了從輸出日志入手.當然,我這里列舉了兩種方式只是給你提供參考,并不是說一共就只有這兩種方式,也不是說,這兩種就是最優(yōu)的.

時序圖

直入主題

有部分朋友反饋說代碼貼圖手機閱讀不友好,但是如果不貼圖的話,很多朋友看完文章自己debug的時候找相應的類和方法又要花費大量時間,所以折中一下,貼圖和貼代碼結合

public ?Invoker refer(Class type, URL url) throws RpcException {

url = url.setProtocol(url.getParameter(Constants.REGISTRY_KEY, Constants.DEFAULT_REGISTRY)).removeParameter(Constants.REGISTRY_KEY);

//序號2,這里的邏輯和之前分享的'zookeeper連接'基本一致,不熟悉的可以回去看看

Registry registry = registryFactory.getRegistry(url);

if (RegistryService.class.equals(type)) {

return proxyFactory.getInvoker((T) registry, type, url);

}

// group="a,b" or group="*"

Map qs = StringUtils.parseQueryString(url.getParameterAndDecoded(Constants.REFER_KEY));

String group = qs.get(Constants.GROUP_KEY);

if (group != null && group.length() > 0 ) {

if ( ( Constants.COMMA_SPLIT_PATTERN.split( group ) ).length > 1

|| "*".equals( group ) ) {

return doRefer( getMergeableCluster(), registry, type, url );

}

}

return doRefer(cluster, registry, type, url);

}

private ?Invoker doRefer(Cluster cluster, Registry registry, Class type, URL url) {

RegistryDirectory directory = new RegistryDirectory(type, url);

directory.setRegistry(registry);

directory.setProtocol(protocol);

URL subscribeUrl = new URL(Constants.CONSUMER_PROTOCOL, NetUtils.getLocalHost(), 0, type.getName(), directory.getUrl().getParameters());

if (! Constants.ANY_VALUE.equals(url.getServiceInterface())

&& url.getParameter(Constants.REGISTER_KEY, true)) {

registry.register(subscribeUrl.addParameters(Constants.CATEGORY_KEY, Constants.CONSUMERS_CATEGORY,

Constants.CHECK_KEY, String.valueOf(false)));

}

//序號3,這里的邏輯和之前分享的'zookeeper訂閱'基本一致,不熟悉的可以回去看看

directory.subscribe(subscribeUrl.addParameter(Constants.CATEGORY_KEY,

Constants.PROVIDERS_CATEGORY

+ "," + Constants.CONFIGURATORS_CATEGORY

+ "," + Constants.ROUTERS_CATEGORY));

//序號4,cluster關鍵字在集群容錯系列也提到過,不熟悉的可以回去看看

return cluster.join(directory);

}

上面的這4步,就完成了schemas標簽信息到invoker的轉換,那么下面就是創(chuàng)建代理對象了(序號5)

private T createProxy(Map map) {

//......(省略部分代碼)

// 創(chuàng)建服務代理

return (T) proxyFactory.getProxy(invoker);

}

我們知道,要封裝這個通信細節(jié),讓用戶像以本地調用方式調用遠程服務,就必須使用代理,然后說到動態(tài)代理,一般我們就想到兩種,一種是JDK的動態(tài)代理,一種是CGLIB的動態(tài)代理,那我們看看兩者有什么特點.

JDK的動態(tài)代理代理的對象必須要實現一個接口,而針對于沒有接口的類,則可用CGLIB.要明白兩者區(qū)別必須要了解原理,之前反復強調,明白了原理自然一通百通.CGLIB其原理也很簡單,對指定的目標類生成一個子類,并覆蓋其中方法實現增強,但由于采用的是繼承,所以不能對final修飾的類進行代理.

除了以上兩種大家都很熟悉的方式外,其實還有一種方式,就是javassist生成字節(jié)碼來實現代理(后面會詳細講,dubbo多處用到了javassist).那dubbo究竟用到了哪種方式實現代理呢?我們往下看

序號5的結束本篇也接近了尾聲.本篇綜合性較強,其中涉及到之前的內容本篇將不再重復提及,可根據注釋中的標記自行查看.

歡迎學Java和大數據的朋友們加入java架構交流: 855835163

加群鏈接:https://jq.qq.com/?_wv=1027&k=5dPqXGI???????

群內提供免費的架構資料還有:Java工程化、高性能及分布式、高性能、深入淺出。高架構。性能調優(yōu)、Spring,MyBatis,Netty源碼分析和大數據等多個知識點高級進階干貨的免費直播講解 ?可以進來一起學習交流哦

?著作權歸作者所有,轉載或內容合作請聯系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容