公司物聯網項目集成Orleans以支持高併發的分佈式業務,對於Orleans也是第一次接觸,本文就分享下我的對Orleans的理解。數據庫
這裏先拋出本身的觀點:Orleans 是一個支持有狀態雲生應用/服務水平伸縮的基於Virtual Actor 模型的.NET分佈式框架。編程
下面我會從如下幾個關鍵點,進行闡述:緩存
在講雲生應用以前,咱們來先講講傳統應用,對於傳統應用經常使用的三層結構以下圖所示。
性能優化
隨着業務的發展,數據庫層一般存在瓶頸,爲了緩解數據庫的壓力,通常會優先考慮增長一層緩存層。服務器
而隨着業務的繼續發展,高併發、大數據量的應用場景逐漸凸顯。若是繼續在單體應用的基礎上進行擴展,能作的無非是增長消息隊列、異步、讀寫分離等機制進行性能優化。整體而言,優化空間不大,但應用的總體複雜度卻隨着引入的新的技術框架而迅速增長,對於應用的維護,是一個潛在的定時炸彈。架構
這個時候你可能會想,既然單體應用單機部署不能知足需求,我能夠作集羣啊。經過將單體應用按照分層結構進行縱向分離,將數據庫從應用服務器分離,將緩存從應用服務器分離。這樣就能夠對分離的各個部分進行分別部署,再借助負載均衡完成集羣效應。到這一步,你的應用應該能撐一段時間了。併發
這個時候,若是回到業務自己去分析,對於一個複雜應用來講,一般的性能瓶頸就是幾個核心服務上。若是可以對存在性能瓶頸的服務進行伸縮,既能大大提升應用的總體可用性又能提升資源的利用率。那怎麼作呢?服務拆分。負載均衡
雲生應用就是服務拆分的結果,雲生應用最大的特色就是:框架
或者簡單來講,雲生應用經過服務拆分支持服務並行,同時各個服務可以快速伸縮以提高系統吞吐量來應對高併發的業務場景。異步
雖然經過服務拆分簡化了整個應用的業務複雜度,可是實現的技術複雜度卻只增不減。
轉向雲生應用咱們面對的第一個難題就是:如何進行服務拆分,才能確保其能分佈式部署,或者說是水平伸縮?!
有經驗的同窗,可能會立馬想到,要將應用/服務設計爲無狀態的。可是這裏我要向你討教幾個問題:
你們不妨先停下來思考一下。(歡迎你們在評論中闡述不一樣觀點。)
這裏,我嘗試從如下兩個角度來談下本身的見解:
1. 對象
面向對象編程強調的是對現實事物的抽象和封裝。經過對事物狀態和行爲進行抽象而後封裝爲對象(類),其中狀態封裝爲類的屬性、字段,將行爲封裝爲類的方法。這個時候獲得的對象是沒有生命力的,由於它本質是一個抽象的結果。只有在程序運行中對類進行實例化獲得一個對象的實例時,才能夠說這個實例對象是有狀態和行爲的,由於這個狀態和行爲是其獨自持有的,這是一個很是核心的條件。獨自持有,換句話說,就是非共享成員。
獨自持有非共享的成員就能夠說這個對象實例是有狀態的嗎?
這裏面你就要看清狀態和有狀態的區別!
舉個簡單例子,大街上你看到一大叔開着豪車,你以爲他很富有。「開着豪車」是你即時看到的狀態屬性。「富有」是你的狀態斷言。但這個狀態斷言是一個假設,畢竟多是借的嘛。
怎樣才能判定「富有」就是這位大叔擁有的狀態呢?很簡單,假設一年365天你每天見到他開豪車,那基本八九不離十了。
因此,若是認定一個對象是否有狀態,還要看其狀態屬性是否持久化!
若是你贊成這個觀點,那麼哪天你看我騎個共享單車,氣喘吁吁從你面前通過,就不要簡單認爲我是苦逼工薪族。畢竟我也是身價上千萬,只是偶爾騎個車鍛鍊鍛鍊。(身價上千萬,昨晚夢到的。)
因此,從對象角度看,一個對象是否有狀態的充分必要條件是:
- 對象已實例化(處於運行時)
- 擁有非共享的狀態屬性
- 狀態持久化
那問題來了,咱們常常寫的類建立的實例,是有狀態的嗎?
2. 應用
基於上面的總結,咱們再來從應用的角度來看分析這個問題。
那應用的狀態和行爲是什麼?
首先,只有運行中的應用纔有狀態和行爲。基於這個前提,我的理解運行時應用的狀態是應用持有的數據,行爲是應用提供的功能。那應用的有無/無狀態界定就要看運行時應用持有的數據可否持久化。
以簡單的Web分層應用舉例 。從邏輯架構上來說應用通常分爲三層,表示層、業務層和數據訪問層。上層進行狀態行爲的封裝,數據層提供數據的持久化。因此從總體的角度來看,其是一個有狀態的應用。但單獨來看,咱們不能對每一層進行有/無狀態的界定。第一,每一層不能單獨運行;第二,分層的目的是爲了職責的隔離,每一層負責相應職責的抽象和封裝,其輸出的是類文件,是對象的集合,沒有生命力。
那從物理架構上來說,Web應用能夠分開兩個部分進行部署:Web實例和MySQL實例。也就是說應用和數據庫是能夠分開部署的。這個時候Web實例就是無狀態的。那咱們通常常說的無狀態服務實際上是就是從這個拆分的角度來講的。
理清完服務拆分的核心問題後,咱們不得不來處理第二個棘手的問題:如何解決雲生應用高併發的應用場景呢?
那首先咱們須要明確處理高併發的難點在哪?第一個是高性能,第二個就是:資源競爭致使的數據一致性問題。對於第一個難點,經過水平擴展服務能夠化解;對於第二個難點,通常就是採用鎖機制,而對於雲生分佈式的應用場景下,處理手段就更加複雜,可能須要使用分佈式鎖,而這種作法,大大下降了應用的總體響應性能。那有沒有更好的解決方案,既兼顧性能又能夠確保數據一致性呢?
有,藉助Actor模型。
簡單來說:Actor模型 = 狀態 + 行爲 + 消息。一個應用/服務由多個Actor組成,每一個Actor都是一個獨立的運行單元,擁有隔離的運行空間,在隔離的空間內,其有獨立的狀態和行爲,不被外界干預,Actor之間經過消息進行交互,而同一時刻,每一個Actor只能被單個線程執行,這樣既有效避免了數據共享和併發問題,又確保了應用的伸縮性。
另外Actor基於事件驅動模型進行異步通訊,性能良好。且位置透明,不管Actor是在本機亦或是在集羣中的其餘機器,均可以直接進行透明調用。
所以Actor模型賦予了應用/服務的生命力(有狀態)、高併發的處理能力和彈性伸縮能力。
對於Actor模型,業界已經有系列的實現框架,好比Erlang、Akka。然而Actor模型做爲一個偏底層的技術框架,對於開發者來講,須要有必定分佈式應用的開發經驗,才能用好Actor(包括Actor的生命週期管理,狀態管理等等)。爲了進一步簡化分佈式編程,微軟的研究人員引入了 Virtual Actor 模型概念,簡單來說Virtual Actor模型是對Actor模型的進一步封裝和抽象。
其與Actor模型的最大的區別在於,Actor的物理實例徹底被抽象出來,並由Virtual Actor所在的運行時自動管理。
Orleans 就是做爲一款面向.NET的Virtual Actor模型的實現框架,提供了開發者友好的編程方式,簡化了分佈式應用的開發成本。在Orleans中Virtual Actor由Grain來體現。
Orleans中核心優點:開發效率高、透明可伸縮。
開發效率高具體表現爲:
透明可伸縮體現爲:
這篇文章,就簡單寫到這裏,對於Orleans的詳細介紹後續會結合實際項目輸出更系統的應用細節,下次再見。