真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

NettyRPC的實現(xiàn)流程

Netty RPC 實現(xiàn)
概念

網(wǎng)站制作、成都網(wǎng)站設計服務團隊是一支充滿著熱情的團隊,執(zhí)著、敏銳、追求更好,是創(chuàng)新互聯(lián)的標準與要求,同時竭誠為客戶提供服務是我們的理念。創(chuàng)新互聯(lián)建站把每個網(wǎng)站當做一個產(chǎn)品來開發(fā),精雕細琢,追求一名工匠心中的細致,我們更用心!

Netty是由JBOSS提供的一個java開源框架,現(xiàn)為 Github上的獨立項目。Netty提供異步的、事件驅(qū)動的網(wǎng)絡應用程序框架和工具,用以快速開發(fā)高性能、高可靠性的網(wǎng)絡服務器和客戶端程序。

也就是說,Netty 是一個基于NIO的客戶、服務器端編程框架,使用Netty 可以確保你快速和簡單的開發(fā)出一個網(wǎng)絡應用,例如實現(xiàn)了某種協(xié)議的客戶、服務端應用。Netty相當于簡化和流線化了網(wǎng)絡應用的編程開發(fā)過程,例如:基于TCP和UDP的socket服務開發(fā)。

"快速"和"簡單"并不用產(chǎn)生維護性或性能上的問題。Netty 是一個吸收了多種協(xié)議(包括FTP、SMTP、HTTP等各種二進制文本協(xié)議)的實現(xiàn)經(jīng)驗,并經(jīng)過相當精心設計的項目。最終,Netty 成功的找到了一種方式,在保證易于開發(fā)的同時還保證了其應用的性能,穩(wěn)定性和伸縮性。

RPC,即 Remote Procedure Call(遠程過程調(diào)用),調(diào)用遠程計算機上的服務,就像調(diào)用本地服務一樣。RPC 可以很好的解耦系統(tǒng),如 WebService 就是一種基于 Http 協(xié)議的 RPC。這個 RPC 整體框架

如下:

Netty RPC的實現(xiàn)流程
關鍵技術

  1. 服務發(fā)布與訂閱:服務端使用 Zookeeper 注冊服務地址,客戶端從 Zookeeper 獲取可用的服務地址。

  2. 通信:使用 Netty 作為通信框架。

  3. Spring:使用 Spring 配置服務,加載 Bean,掃描注解。

  4. 動態(tài)代理:客戶端使用代理模式透明化服務調(diào)用。

  5. 消息編解碼:使用 Protostuff 序列化和反序列化消息。

核心流程

  1. 服務消費方(client)調(diào)用以本地調(diào)用方式調(diào)用服務;

  2. client stub 接收到調(diào)用后負責將方法、參數(shù)等組裝成能夠進行網(wǎng)絡傳輸?shù)南Ⅲw;

  3. client stub 找到服務地址,并將消息發(fā)送到服務端;

  4. server stub 收到消息后進行解碼;

  5. server stub 根據(jù)解碼結果調(diào)用本地的服務;

  6. 本地服務執(zhí)行并將結果返回給 server stub;

  7. server stub 將返回結果打包成消息并發(fā)送至消費方;

  8. client stub 接收到消息,并進行解碼;

  9. 服務消費方得到最終結果。

RPC 的目標就是要 2~8 這些步驟都封裝起來,讓用戶對這些細節(jié)透明。JAVA 一般使用動態(tài)代理方式實現(xiàn)遠程調(diào)用。

Netty RPC的實現(xiàn)流程
消息編解碼

息數(shù)據(jù)結構(接口名稱+方法名+參數(shù)類型和參數(shù)值+超時時間+ requestID)
客戶端的請求消息結構一般需要包括以下內(nèi)容:

  1. 接口名稱:在我們的例子里接口名是“HelloWorldService”,如果不傳,服務端就不知道調(diào)用哪個接口了;

  2. 方法名:一個接口內(nèi)可能有很多方法,如果不傳方法名服務端也就不知道調(diào)用哪個方法;

  3. 參數(shù)類型和參數(shù)值:參數(shù)類型有很多,比如有 bool、int、long、double、string、map、list,甚至如 struct(class);以及相應的參數(shù)值;

  4. 超時時間:

  5. requestID,標識唯一請求 id,在下面一節(jié)會詳細描述 requestID 的用處。

  6. 服務端返回的消息 : 一般包括以下內(nèi)容。返回值+狀態(tài) code+requestID

序列化

目前互聯(lián)網(wǎng)公司廣泛使用 Protobuf、Thrift、Avro 等成熟的序列化解決方案來搭建 RPC 框架,這些都是久經(jīng)考驗的解決方案。

通訊過程

核心問題(線程暫停、消息亂序)

如果使用 netty 的話,一般會用 channel.writeAndFlush()方法來發(fā)送消息二進制串,這個方法調(diào)用后對于整個遠程調(diào)用(從發(fā)出請求到接收到結果)來說是一個異步的,即對于當前線程來說,將請求發(fā)送出來后,線程就可以往后執(zhí)行了,至于服務端的結果,是服務端處理完成后,再以消息的形式發(fā)送給客戶端的。于是這里出現(xiàn)以下兩個問題:

  1. 怎么讓當前線程“暫?!保冉Y果回來后,再向后執(zhí)行?

  2. 如果有多個線程同時進行遠程方法調(diào)用,這時建立在 client server 之間的 socket 連接上會有很多雙方發(fā)送的消息傳遞,前后順序也可能是隨機的,server 處理完結果后,將結果消息發(fā)送給 client,client 收到很多消息,怎么知道哪個消息結果是原先哪個線程調(diào)用的?如下圖所示,線程 A 和線程 B 同時向 client socket 發(fā)送請求 requestA 和 requestB,socket 先后將 requestB 和 requestA 發(fā)送至 server,而 server 可能將 responseB 先返回,盡管 requestB 請求到達時間更晚。我們需要一種機制保證 responseA 丟給****ThreadA,responseB 丟給 ThreadB。

Netty RPC的實現(xiàn)流程
通訊流程

requestID 生成-AtomicLong

  1. client 線程每次通過 socket 調(diào)用一次遠程接口前,生成一個唯一的 ID,即 requestID(requestID 必需保證在一個 Socket 連接里面是唯一的),一般常常使用 AtomicLong從 0 開始累計數(shù)字生成唯一 ID;存放回調(diào)對象 callback 到全局 ConcurrentHashMap

  2. 將 處 理 結 果 的 回 調(diào) 對 象 callback , 存 放 到 全 局 ConcurrentHashMap 里 面put(requestID, callback);synchronized 獲取回調(diào)對象 callback 的鎖并自旋 wait

  3. 當線程調(diào)用 channel.writeAndFlush()發(fā)送消息后,緊接著執(zhí)行 callback 的 get()方法試圖獲取遠程返回的結果。在 get()內(nèi)部,則使用 synchronized 獲取回調(diào)對象 callback 的鎖,再先檢測是否已經(jīng)獲取到結果,如果沒有,然后調(diào)用 callback 的 wait()方法,釋放callback 上的鎖,讓當前線程處于等待狀態(tài)。監(jiān)聽消息的線程收到消息,找到 callback 上的鎖并喚醒

  4. 服務端接收到請求并處理后,將 response 結果(此結果中包含了前面的 requestID)發(fā)送給客戶端,客戶端 socket 連接上專門監(jiān)聽消息的線程收到消息,分析結果,取到requestID , 再 從 前 面 的 ConcurrentHashMap 里 面 get(requestID) , 從 而 找 到callback 對象,再用 synchronized 獲取 callback 上的鎖,將方法調(diào)用結果設置到callback 對象里,再調(diào)用 callback.notifyAll()喚醒前面處于等待狀態(tài)的線程。
    Netty RPC的實現(xiàn)流程
    未完待續(xù)…………

提前一飽眼福,附完整 JVM pdf文檔和最新學習視頻
VX獲取:13272413561
Netty RPC的實現(xiàn)流程
Netty RPC的實現(xiàn)流程
Netty RPC的實現(xiàn)流程


本文名稱:NettyRPC的實現(xiàn)流程
瀏覽地址:http://weahome.cn/article/ghddei.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部