Dubbo實現源碼分析


1. Dubbo概述web

Dubbo是阿里巴巴開源出來的一個分佈式服務框架,致力於提供高性能和透明化的RPC遠程服務調用方案,以及做爲SOA服務治理的方案。它的核心功能包括:sql

#remoting:遠程通信基礎,提供對多種NIO框架抽象封裝,包括「同步轉異步」和「請求-響應」模式的信息交換方式。 #Cluster: 服務框架核心,提供基於接口方法的遠程過程調用,包括多協議支持,並提供軟負載均衡和容錯機制的集羣支持。 #registry: 服務註冊中心,使服務消費方能動態的查找服務提供方,使地址透明,使服務提供方能夠平滑增長或減小機器。網絡

這裏咱們只是補充一下從源碼具體實現角度來看的某些細節方面,包括Invoker、ExtensionLoader等方面。任何官方已經介紹過的細節,咱們不作多此一舉,官方文檔已經足夠詳實了,這篇文檔的定位是補充實現的相關細節,是基於我在往Dubbo添加web service協議過程當中,所碰到過的一些困難。架構

2. 服務提供者暴露一個服務的詳細過程併發

上圖是服務提供者暴露服務的主過程:app

首先ServiceConfig類拿到對外提供服務的實際類ref(如:HelloWorldImpl),而後經過ProxyFactory類的getInvoker方法使用ref生成一個AbstractProxyInvoker實例,到這一步就完成具體服務到Invoker的轉化。接下來就是Invoker轉換到Exporter的過程。負載均衡

Dubbo處理服務暴露的關鍵就在Invoker轉換到Exporter的過程(如上圖中的紅色部分),下面咱們以Dubbo和RMI這兩種典型協議的實現來進行說明:框架

#Dubbo的實現異步

Dubbo協議的Invoker轉爲Exporter發生在DubboProtocol類的export方法,它主要是打開socket偵聽服務,並接收客戶端發來的各類請求,通信細節由Dubbo本身實現。jvm

#RMI的實現

RMI協議的Invoker轉爲Exporter發生在RmiProtocol類的export方法,它經過Spring或Dubbo或JDK來實現RMI服務,通信細節這一塊由JDK底層來實現,這就省了很多工做量。

3. 服務消費者消費一個服務的詳細過程

上圖是服務消費的主過程:

首先ReferenceConfig類的init方法調用Protocol的refer方法生成Invoker實例(如上圖中的紅色部分),這是服務消費的關鍵。接下來把Invoker轉換爲客戶端須要的接口(如:HelloWorld)。

關於每種協議如RMI/Dubbo/Web service等它們在調用refer方法生成Invoker實例的細節和上一章節所描述的相似。

 

4. 滿眼都是Invoker

因爲Invoker是Dubbo領域模型中很是重要的一個概念,不少設計思路都是向它靠攏。這就使得Invoker滲透在整個實現代碼裏,對於剛開始接觸Dubbo的人,確實容易給搞混了。

下面咱們用一個精簡的圖來講明最重要的兩種Invoker:服務提供Invoker和服務消費Invoker:

爲了更好的解釋上面這張圖,咱們結合服務消費和提供者的代碼示例來進行說明:

#服務消費者代碼

#服務消費者代碼 public class DemoClientAction { private DemoService demoService; public void setDemoService(DemoService demoService) { this.demoService = demoService; } public void start() { String hello = demoService.sayHello("world" + i); } }

上面代碼中的’DemoService’就是上圖中服務消費端的proxy,用戶代碼經過這個proxy調用其對應的Invoker(DubboInvoker、 HessianRpcInvoker、 InjvmInvoker、 RmiInvoker、 WebServiceInvoker中的任何一個),而該Invoker實現了真正的遠程服務調用。

#服務提供者代碼 public class DemoServiceImpl implements DemoService { public String sayHello(String name) throws RemoteException { return "Hello " + name; } }

上面這個類會被封裝成爲一個AbstractProxyInvoker實例,並新生成一個Exporter實例。這樣當網絡通信層收到一個請求後,會找到對應的Exporter實例,並調用它所對應的AbstractProxyInvoker實例,從而真正調用了服務提供者的代碼。

Dubbo裏還有一些其餘的Invoker類,但上面兩種是最重要的。

5. ExtensionLoader的完整分析

ExtensionLoader是Dubbo中一個很是重要的類,剛接觸Dubbo源碼的人看這個類的時候也多少會有點困惑,這個類很是重要,它就像是廚房裏的「大廚」,按照用戶的隨時須要把各類「食材」烹調出來。

咱們結合具體代碼詳細說一下ExtensionLoader的實現,下面是ServiceConfig類裏的一行代碼:

private static final Protocol protocol = ExtensionLoader.getExtensionLoader(Protocol.class).getAdaptiveExtension();

上面代碼的程序流程圖以下所示(假定是第一次執行這行代碼):

在這個過程當中最重要的兩個方法是getExtensionClasses和createAdaptiveExtensionClass(圖中紅色部分),下面詳細對這兩個方法進行分析:

#getExtensionClasses

這個方法主要讀取META-INF/services/目錄下對應文件內容,在本示例代碼中,是讀取META-INF/services/com.alibaba.dubbo.rpc.Protocol文件中的內容,具體內容以下:

com.alibaba.dubbo.registry.support.RegistryProtocol com.alibaba.dubbo.rpc.protocol.ProtocolFilterWrapper com.alibaba.dubbo.rpc.protocol.ProtocolListenerWrapper com.alibaba.dubbo.rpc.protocol.dubbo.DubboProtocol com.alibaba.dubbo.rpc.protocol.injvm.InjvmProtocol com.alibaba.dubbo.rpc.protocol.rmi.RmiProtocol com.alibaba.dubbo.rpc.protocol.hessian.HessianProtocol com.alibaba.dubbo.rpc.protocol.webservice.WebServiceProtocol

它分析該文件中的每一行(每一行對應一個類),分析這些類,若是發現有哪一個類的Annotation是@Adaptive,則找到對應的AdaptiveClass了,但因爲Protocol文件裏沒有哪一個類的Annotation是@Adaptive,因此在這個例子中該方法沒找到對應的AdaptiveClass。

#createAdaptiveExtensionClass

該方法是在getExtensionClasses方法找不到AdaptiveClass的狀況下被調用,該方法主要是經過字節碼的方式在內存中新生成一個類,它具備AdaptiveClass的功能,Protocol就是經過這種方式得到AdaptiveClass類的。

歡迎工做一到五年的Java工程師朋友們加入Java架構開發: 855835163 羣內提供免費的Java架構學習資料(裏面有高可用、高併發、高性能及分佈式、Jvm性能調優、Spring源碼,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多個知識點的架構資料)合理利用本身每一分每一秒的時間來學習提高本身,不要再用"沒有時間「來掩飾本身思想上的懶惰!趁年輕,使勁拼,給將來的本身一個交代!

相關文章
相關標籤/搜索