微服務架構的核心要點和實現原理解析

摘要:本文中,咱們將進一步理解微服務架構的核心要點和實現原理,爲讀者的實踐提供微服務的設計模式,以期讓微服務在讀者正在工做的項目中起到積極的做用。前端

微服務架構中職能團隊的劃分

傳統單體架構將系統分紅具備不一樣職責的層次,對應的項目管理也傾向於將大的團隊分紅不一樣的職能團隊,主要包括:用戶交互UI團隊、後臺業務邏輯處理團隊與數據存取ORM團隊、DBA團隊等。每一個團隊只對本身分層的職責負責,並對使用方提供組件服務質量保證。若是其中一個模塊化組件須要升級、更新,那麼這個變動會涉及不一樣的分層團隊,即便升級和變動的改變很小,也須要進行跨團隊溝通:需求階段須要跨團隊溝通產品功能,設計階段須要跨團隊溝通設計方案,開發階段須要跨團隊溝通具體的接口定義,測試階段須要溝通業務迴歸等事宜,甚至上線都須要跨團隊溝通應用的上線順序。可見在傳統的總體架構下,後期的維護成本很高,出現事故的風險很大。數據庫

根據康威定律,團隊的交流機制應該與架構設計機制相對應。所以,在微服務架構下,職能團隊的劃分方法是咱們首先要考慮的一個核心要素。後端

微服務時代的團隊溝通方式如圖1-1所示。設計模式


圖1-1

微服務架構按照業務的功能進行劃分,每一個單一的業務功能叫做一個服務,每一個服務對應一個獨立的職能團隊,團隊裏包含用戶交互UI設計師、後臺服務開發人員、DBA、運營和運維人員。數組

在傳統的總體架構中,軟件是有生命週期的,經歷需求分析、開發和測試,而後被交付給運維團隊,這時開發團隊將會解散,這是對軟件的一個「放手」。而在微服務架構中,提倡運維人員也是服務項目團隊的一員,倡導誰開發、誰維護,實施終生維護制度。緩存

在業務服務的內部實現須要升級或者變動時,團隊內的各角色成員進行溝通便可,而不須要進行跨團隊溝通,這大大提升了溝通效率。只有服務之間的接口須要變動時才須要跨部門溝通,若是前期在服務之間的交互上定義了良好的接口,則接口變動的機率並不大,即便接口模式有變動,也能夠經過必定的設計模式和規範來解決。安全

微服務的去中心化治理

筆者曾經在一個互聯網平臺上工做,平臺的決策者倡導建設API網關,全部外部服務和內部服務都由統一的API網關進行管理。在項目初期,中心化的API網關統一了全部API的入口,這看起來很規範,但從技術角度來看限制了API的多樣化。隨着業務的發展,API網關開始暴露問題,每一個用戶請求通過機房時只要有服務之間的交互,則都會從API網關進行路由,服務上量之後,因爲內部服務之間的交互都會疊加在API網關的調用上,因此在很大程度上放大了API網關的調用TPS,API網關很快就遇到了性能瓶頸。性能優化

這個案例是典型的微服務的反模式,微服務倡導去中心化的治理,不推薦每一個微服務都使用相同的標準和技術來開發和使用服務。在微服務架構下可使用C++開發一個服務,來對接Java開發的另一個服務,對於異構系統之間的交互標準,一般可使用工具來補償。開發者能夠開發共用的工具,並分享給異構系統的開發者使用,來解決異構系統不一致的問題,例如:Thrift遠程調用框架使用中間語言(IDL)來定義接口,中間語言是獨立於任何語言的,並提供了工具來生成中間語言,以及在中間語言與具體語言之間的代碼轉換。bash

微服務架構倡導去中心化的服務管理和治理,儘可能不設置中心化的管理服務,最差也須要在中心化的管理服務宕機時有替代方案和設計。在筆者工做的支付平臺服務化建設中,第1層SOA服務化採用Dubbo框架進行定製化,若是Dubbo服務化出現了大面積的崩潰,則服務化體系會切換到點對點的hessian遠程調用,這被稱爲服務化降級,降級後點對點的hessian遠程調用時沒有中心化節點,總體上符合微服務的原理。網絡

微服務的交互模式

本節介紹微服務之間交互的通用設計模式,這些設計模式對微服務之間的交互定義契約,服務的生產者和調用者都須要遵照這些契約,才能保證微服務不出問題。

1. 讀者容錯模式

讀者容錯模式(Tolerant Reader)指微服務化中服務提供者和消費者之間如何對接口的改變進行容錯。從字面上來說,消費者須要對提供者提供的功能進行兼容性設計,尤爲對服務提供者返回的內容進行兼容,或者解決在服務提供者改變接口或者數據的格式的狀況下,如何讓服務消費者正常運行。

任何一個產品在設計時都沒法預見未來可能增長的全部需求,服務的開發者一般經過迭代及時地增長新功能,或者讓服務提供的API天然地演進。不過,服務提供者對外提供的接口的數據格式的改變、增長和刪除,都會致使服務的消費者不能正常工做。

所以,在服務消費者處理服務提供者返回的消息的過程當中,須要對服務返回的消息進行過濾,只提取本身須要的內容,對多餘或者未知的內容採起拋棄的策略,而不是硬生生地拋錯處理。

在實現過程當中不推薦使用嚴格的校驗策略,而是推薦使用寬鬆的校驗策略,即便服務消費者拿到的消息報文發生了改變,程序也只需盡最大努力提取須要的數據,同時忽略不可識別的數據。只有在服務消費者徹底不能識別接收到的消息,或者沒法經過識別的信息繼續處理流程時,才能拋出異常。

服務的消費者的容錯模式忽略了新的消息項、可選的消息項、未知的數據值及服務消費者不須要的數據項。

筆者當前在某個支付公司工做,公司裏幾乎每一個業務都須要使用枚舉類型,在微服務平臺下,筆者在研發流程規範中定義了一條枚舉值使用規範:

在服務接口的定義中,參數可使用枚舉值,在返回值的DTO中禁止使用枚舉值。

這條規範就是讀者容錯模式在實踐中的一個實例,之因此在參數中容許使用枚舉值,是由於若是服務的提供者升級了接口,增長了枚舉值,若服務的消費者並無感知,則服務的消費者得知新的枚舉值就能夠傳遞新的枚舉值了;可是若是接口的返回DTO中使用了枚舉值,而且由於某種緣由增長了枚舉值,則服務消費者在反序列化枚舉時就會報錯,所以在返回值中咱們應該使用字符串等互相承認的類型,來作到雙方的互相兼容,並實現讀者容錯模式。

2. 消費者驅動契約模式

消費者驅動契約模式用來定義服務化中服務之間交互接口改變的最佳規則。

服務契約分爲:提供者契約、消費者契約及消費者驅動的契約,它從指望與約束的角度描述了服務提供者與服務消費者之間的聯動關係。

  • 提供者契約:是咱們最經常使用的一種服務契約,顧名思義,提供者契約是以提供者爲中心的,提供者提供了什麼功能和消息格式,各消費者都會無條件地遵照這些約定,不論消費者實際須要多少功能,消費者接受了提供者契約時,都會根據服務提供者的規則來使用服務。

  • 消費者契約:是對某個消費者的需求進行更爲精確的描述,在一次具體的服務交互場景下,表明消費者須要提供者提供功能中的哪部分數據。消費者契約能夠被用來標識現有的提供者契約,也能夠用來發現一個還沒有明確的提供者契約。

  • 消費者驅動的契約:表明服務提供者向其全部當前消費者承諾遵照的約束。一旦各消費者把本身的具體指望告知提供者,則提供者不管在什麼時間和場景下,都不該該打破契約。

在現實的服務交互設計中,上面這三種契約是同時存在的,筆者所在的支付平臺裏,交易系統在完成一筆支付後,須要到帳務系統爲商戶入帳,在這個過程當中,服務契約表現以下。

  • 生產者契約:帳務系統提供Dubbo服務化接口,參數爲商戶帳戶ID、入帳訂單號和入帳金額。

  • 消費者契約:帳務系統返回DTO,包含商戶帳戶ID、入帳訂單號、入帳金額、入帳時間、帳務流水號、入帳狀態等,而交易系統只需使用其中的入帳訂單號和入帳狀態。

  • 消費者驅動的契約:爲了保證資金安全,交易系統做爲入帳的發起者向帳務提出要求,須要帳務作冪等和濾重處理,對重複的入帳請求進行攔截;帳務系統在接受這個契約後,即便未來有任何改變,也不能打破這個限制,不然就會形成資金的損失,這在金融系統中是最嚴重的問題。

服務之間的交互須要使用的三種服務契約如圖1-2所示。


圖1-2

從圖1-2能夠看到,服務提供者契約是服務提供者單方面定下的規則,而一個消費者契約會成爲提供者契約的一部分,多個服務消費者能夠對服務提供者提出約束,服務提供者須要在未來遵照服務消費者提出的契約,這就是消費者驅動的契約。

3. 去數據共享模式

與SOA服務化對比,微服務是去ESB總線、去中心化及分佈式的;而SOA仍是以ESB爲核心實現遺留系統的集成,以及基於Web Service爲標準實現的通用的面向服務的架構。在微服務領域,微服務之間的交互經過定義良好的接口來實現,不容許使用共享數據來實現。

在實踐的過程當中,有些方案的設計使用緩存或者數據庫做爲兩個微服務之間的紐帶,在業務流程的處理過程當中,爲了處理簡單,前一個服務將中間結果存入數據庫或者緩存,下一個服務從緩存或者數據庫中拿出數據繼續處理。處理流程如圖1-3所示。


圖1-3

這種交互流程的缺點以下。

  • 使得微服務之間的交互除了接口契約,還存在數據存儲契約。
  • 上游的數據格式發生變化時,可能致使下游的處理邏輯出現問題。
  • 多個服務共享一個資源服務,對資源服務的運維難以劃清職責和界限。
  • 在作雙機房獨立部署時,須要考慮服務和資源的路由狀況,跨機房的服務調用不能使用獨立的資源部署模式,所以難以實現服務自治。

所以,在設計微服務架構時,必定不要共享緩存和數據庫等資源,也不要使用總線模式,服務之間的通訊和交互只能依賴定義良好的接口,一般使用RESTful樣式的API或者透明的RPC調用框架。

微服務的分解和組合模式

使用微服務架構劃分服務和團隊是微服務架構實施的重要一步,良好的劃分和拆分使系統達到鬆耦合和高內聚的效果,而後經過微服務的靈活組裝能夠知足上層的各類各樣的業務處理需求。

在微服務架構的需求分析和架構設計過程當中,一般是用領域的動詞和名詞來劃分微服務的,例如,對於一個電商後臺系統,能夠分解爲訂單、商品、商品目錄、庫存、購物車、交易、支付、發票、物流等子系統,每一個名詞和動詞均可以是一個微服務,將這幾個微服務組合在一塊兒,就實現了電商平臺用戶購買商品的整個業務流。

這樣拆分之後,系統具備敏捷性、靈活性、可伸縮性等,拆分後有多個高度自治的微服務,那麼以什麼方式組合微服務呢?

1. 服務代理模式

服務代理模式是最簡單的服務組合模式,它根據業務的需求選擇調用後端的某個服務。在返回給使用端以前,代理能夠對後端服務的輸出進行加工,也能夠直接把後端服務的返回結果返回給使用端。

服務代理模式的架構如圖1-4所示。


圖1-4

在筆者工做的微服務化架構平臺下,常常會使用這種模式,典型的案例是作平滑的系統遷移,一般經歷以下4個階段。

  • 在新老系統上雙寫。
  • 遷移雙寫以前的歷史遺留數據。
  • 將讀請求切換到新系統。
  • 下調雙寫邏輯,只寫新系統。

服務代理模式經常應用到第3步,通常會對讀請求切換設計一個開關,開關打開時查詢新系統,開關關閉時查詢老系統。

遷移案例中開關的邏輯如圖1-5所示。


圖1-5

2. 服務聚合模式

服務聚合模式是最經常使用的服務組合模式,它根據業務流程處理的須要,以必定的順序調用依賴的多個微服務,對依賴的微服務返回的數據進行組合、加工和轉換,最後以必定的形式返回給使用方。

這裏,每一個被依賴的微服務都有本身的緩存和數據庫,聚合服務自己能夠有本身的數據存儲,包括緩存和數據庫等,也能夠是簡單的聚合,不須要持久化任何數據。

服務聚合模式的架構如圖1-6所示。


圖1-6

這裏體現了DRY(Don’t Repeat Yourself)原則的設計理念,在設計或者構造應用時,最大限度地重用了現有的實現。假如一塊業務邏輯由三個獨立的邏輯塊組成,每一個獨立的邏輯塊可能有多個使用方,則DRY原則推薦將三個獨立的邏輯塊封裝成三個獨立運行的微服務,而後使用本節的服務聚合模式開發聚合服務,將三個獨立的邏輯塊聚合在一塊兒提供給上層組合服務。這樣的設計原則有以下好處。

  • 三個獨立的子服務能夠各自獨立開發、敏捷變動和部署。
  • 聚合服務封裝下層的業務處理服務,由三個獨立的子服務完成數據持久化等工做,項目結構清晰明瞭。
  • 三個獨立的子服務對於其餘使用方仍然能夠重用。

考慮到本節開頭的例子,在對微服務進行拆分時,將電商後臺系統大體拆分紅訂單、商品、商品目錄、庫存、購物車、交易、支付、發票、物流等微服務,那麼電商平臺的前端應用就是後端各個微服務的一個最大的聚合服務,前端應用經過調用商品和商品目錄顯示商品列表,提供給用戶選擇商品的功能,用戶選擇商品後增長商品到購物車,在用戶從購物車結算時,調用交易系統完成交易和支付等。

電商前臺的聚合模式的案例架構如圖1-7所示。


圖1-7

另外,聚合服務也能夠是一個純後臺服務,經過聚合對使用方輸出組合的服務,例如在上面的電商系統案例中,在用戶選擇結算後,系統調用交易,交易系統會調用庫存系統鎖庫存,而後建立交易訂單,引導用戶去支付,支付成功後扣減庫存,最後經過發票服務開具電子發票。

電商後臺交易服務的聚合模式架構如圖1-8所示。


圖1-8

3. 服務串聯模式

服務串聯模式相似於一個工做流,最前面的服務1負責接收請求和響應使用方,串聯服務後再與服務1交互,隨後服務1與服務2交互,最後,從服務2產生的結果通過服務1和串聯服務逐個處理後返回給使用方,如圖1-9所示。


圖1-9

服務串聯模式之間的調用一般使用同步的RESTful風格的遠程調用實現,注意,這種模式採用的是同步調用方式,在串聯服務沒有完成並返回以前,全部服務都會阻塞和等待,一個請求會佔用一個線程來處理,所以在這種模式下不建議服務的層級太多,若是能用服務聚合模式代替,則優先使用服務聚合模式,而不是使用這種服務串聯模式。

相對於服務聚合模式,服務串聯模式有一個優勢,即串聯鏈路上再增長一個節點時,只要不是在串聯服務的正後面增長,那麼串聯服務是無感知的。

在串聯服務中調用鏈的最後端增長服務無感知的架構如圖1-10所示。


圖1-10

在上面說起的電商案例中,UI前端應用調用交易,交易調用商品庫存系統鎖定庫存和扣減庫存,使用的就是服務串聯模式。

服務串聯模式案例的架構如圖1-11所示。


圖1-11

4. 服務分支模式

服務分支模式是服務代理模式、服務聚合模式和服務串聯模式相結合的產物。

分支服務能夠擁有本身的數據庫存儲,調用多個後端的服務或者服務串聯鏈,而後將結果進行組合處理再返回給客戶端。分支服務也可使用代理模式,簡單地調用後端的某個服務或者服務鏈,而後將返回的數據直接返回給使用方。

服務分支模式的架構如圖1-12所示。


圖1-12

在實際的業務平臺建設中,因爲業務的複雜性,抽象的微服務可能有多層的依賴關係,依賴關係並不會太簡單,常常呈現樹形的分支結構。

以電商平臺的支付服務架構爲例,如圖1-13所示。


圖1-13

支付服務對接兩個外部的支付網關,都要通過各自的支付渠道網關,同時支持帳戶餘額支付,這個支付服務其實就是一個分支模式,在實際項目中這種服務分支模式不少。

筆者在構建支付平臺時,因爲大量地使用了服務分支模式,因此發現了一個比較有趣的現象,以下所述。

假設有一個基礎服務,在服務分支模式的多個層次中對基礎服務都有依賴,那麼當基礎服務的一臺機器宕機時,假設基礎服務有8臺機器,則最後受影響的流量並非1/8。假設基礎服務6共有8臺機器,服務一、服務3和服務5組成某服務的一個調用鏈,則調用鏈過程當中會屢次調用基礎服務6。

具體服務的調用鏈示意圖如圖1-14所示。


圖1-14

某天,基礎服務6的8臺機器中的1臺宕機,按照常理,你們都認爲隻影響其中1/8的流量,而統計結果顯示影響的業務結果居然大於1/8。

仔細思考,形成這個結果的緣由是調用鏈上有多個層次重複調用了基礎服務,致使基礎服務掛掉時影響的流量有累加效果,具體計算以下。

假設進入系統的流量爲n,調用鏈從服務3開始調用服務6,服務3有1/8的流量失敗,這時剩下的成功的流量爲7/8 ×n,剩下的成功的流量繼續走到服務5,服務5再次調用服務6,又有1/8的流量失敗,剩下7/8 × 7/8× n。

假設基礎服務資源池中的機器個數爲i,一次掛掉的機器個數爲j,一個調用鏈中調用x次基礎服務,那麼正確處理的流量的計算公式爲:


假設容許的可用性波動率爲a,求出底層服務一次宕機1臺時最少應該配置的機器數爲:


對公式進行轉換:


因爲一次只容許一臺機器宕機:


因此得出須要設置的機器數量i爲:


對於上面的案例,每次最多容許基礎服務6宕機1臺,在這種狀況下須要保持可用性的波動率小於25%,一共有兩層服務依賴基礎服務6,經過上述公式計算得出:

i > 7.5

結果,至少爲服務6部署9臺機器,這樣在1臺機器宕機時,對可用性的波動性影響控制在25%之內。

因爲分支模式放大了服務的依賴關係,所以在現實的微服務設計中儘可能保持服務調用級別的簡單,在使用服務組合和服務代理模式時,不要使用服務串聯模式和服務分支模式,以保持服務依賴關係的清晰明瞭,這也減小了往後維護的工做量。

5. 服務異步消息模式

前面的全部服務組合模式都使用同步的RESTful風格的同步調用來實現,同步調用模式在調用的過程當中會阻塞線程,若是服務提供方遲遲沒有返回,則服務消費方會一直阻塞,在嚴重狀況下會撐滿服務的線程池,出現雪崩效應。

所以,在構建微服務架構系統時,一般會梳理核心系統的最小化服務集合,這些核心的系統服務使用同步調用,而其餘核心鏈路之外的服務可使用異步消息隊列進行異步化。

服務異步消息模式的架構如圖1-15所示。


圖1-15

在圖1-15中,聚合服務同步調用服務1和服務2,而服務2經過消息隊列將異步消息傳遞給服務3和服務4。

典型的案例就是在電商系統中,交易完成後向物流系統發起消息通知,通知物流系統發貨,如圖1-16所示。


圖1-16

6. 服務共享數據模式

服務共享數據模式實際上是反模式,在1.3.3節中提出了去數據共享模式,因爲去掉了數據共享,因此僅僅經過服務之間良好定義的接口進行交互和通訊,使得每一個服務都是自治的,服務自己和服務的團隊包含全角色棧的技術和運營人員,這些人都是專業的人作專業的事,使溝通在團隊內部解決,所以可使效率最大化。

服務共享數據模式的架構如圖1-17所示。


圖1-17

然而,在下面兩種場景下,咱們仍然須要數據共享模式。

  • 單元化架構

一些平臺因爲對性能有較高的要求,因此採用微服務化將服務進行拆分,經過網絡服務進行通訊,儘管網絡通訊的帶寬已經很寬,可是還會有性能方面的損耗,在這種場景下,可讓不一樣的微服務共享一些資源,例如:緩存、數據庫等,甚至能夠將緩存和數據在物理拓撲上與微服務部署在一個物理機中,最大限度地減小網絡通訊帶來的性能損耗,咱們將這種方法稱爲「單元化架構」。

單元化架構的示意圖如圖1-18所示。


圖1-18

  • 遺留的總體服務

對於歷史遺留的傳統單體服務,咱們在重構微服務的過程當中,發現單體服務依賴的數據庫表耦合在一塊兒,對其拆分須要進行反規範化的處理,可能會形成數據一致性問題,在沒有對其徹底理解和有把握的前提下,會選擇保持現狀,讓不一樣的微服務暫時共享數據存儲。

除了上面提到的兩個場景,任何場景都不能使用服務數據共享模式。

微服務的容錯模式

在使用了微服務架構之後,總體的業務流程被拆分紅小的微服務,並組合在一塊兒對外提供服務,微服務之間使用輕量級的網絡協議通訊,一般是RESTful風格的遠程調用。因爲服務與服務的調用再也不是進程內的調用,而是經過網絡進行的遠程調用,衆所周知,網絡通訊是不穩定、不可靠的,一個服務依賴的服務可能出錯、超時或者宕機,若是沒有及時發現和隔離問題,或者在設計中沒有考慮如何應對這樣的問題,那麼極可能在短期內服務的線程池中的線程被用滿、資源耗盡,致使出現雪崩效應。本節針對微服務架構中可能遇到的這些問題,講解應該採起哪些措施和方案來解決。

1. 艙壁隔離模式

這裏用航船的設計比喻艙壁隔離模式,若一艘航船遇到了意外事故,其中一個船艙進了水,則咱們但願這個船艙和其餘船艙是隔離的,但願其餘船艙能夠不進水,不受影響。在微服務架構中,這主要體如今以下兩個方面。

1)微服務容器分組

筆者所在的支付平臺應用了微服務,將微服務的每一個節點的服務池分爲三組:準生產環境、灰度環境和生產環境。準生產環境供內側使用;灰度環境會跑一些普通商戶的流量;大部分生產流量和VIP商戶的流量則跑在生產環境中。這樣,在一次比較大的重構過程當中,咱們就能夠充分利用灰度環境的隔離性進行預驗證,用普通商戶的流量驗證重構沒有問題後,再上生產環境。

另一個案例是一些社交平臺將名人的自媒體流量所有路由到服務的核心池子中,而將普通用戶的流量路由到另一個服務池子中,有效隔離了普通用戶和重要用戶的負載。

其服務分組如圖1-19所示。


圖1-19

2)線程池隔離

在微服務架構實施的過程當中,咱們不必定將每一個服務拆分到微小的力度,這取決於職能團隊和財務的情況,咱們通常會將同一類功能劃分在一個微服務中,儘可能避免微服務過細而致使成本增長,適可而止。

這樣就會致使多個功能混合部署在一個微服務實例中,這些微服務的不一樣功能一般使用同一個線程池,致使一個功能流量增長時耗盡線程池的線程,而阻塞其餘功能的服務。

線程池隔離如圖1-20所示。


圖1-20

2. 熔斷模式

能夠用家裏的電路保險開關來比喻熔斷模式,若是家裏的用電量過大,則電路保險開關就會自動跳閘,這時須要人工找到用電量過大的電器來解決問題,而後打開電路保險開關。在這個過程當中,電路保險開關起到保護整個家庭電路系統的做用。

對於微服務系統也同樣,當服務的輸入負載迅速增長時,若是沒有有效的措施對負載進行熔斷,則會使服務迅速被壓垮,服務被壓垮會致使依賴的服務都被壓垮,出現雪崩效應,所以,可經過模擬家庭的電路保險開關,在微服務架構中實現熔斷模式。

微服務化的熔斷模式的狀態流轉如圖1-21所示。


圖1-21

3. 限流模式

服務的容量和性能是有限的,在第3章中會介紹如何在架構設計過程當中評估服務的最大性能和容量,然而,即便咱們在設計階段考慮到了性能壓力的問題,並從設計和部署上解決了這些問題,可是業務量是隨着時間的推移而增加的,忽然上量對於一個飛速發展的平臺來講是很常見的事情。

針對服務忽然上量,咱們必須有限流機制,限流機制通常會控制訪問的併發量,例如每秒容許處理的併發用戶數及查詢量、請求量等。

有以下幾種主流的方法實現限流。

1)計數器

經過原子變量計算單位時間內的訪問次數,若是超出某個閾值,則拒絕後續的請求,等到下一個單位時間再從新計數。

在計數器的實現方法中一般定義了一個循環數組(見圖1-22),例如:定義5個元素的環形數組,計數週期爲1s,能夠記錄4s內的訪問量,其中有1個元素爲當前時間點的標誌,一般來講每秒程序都會將前面3s的訪問量打印到日誌,供統計分析。


圖1-22

咱們將時間的秒數除以數組元素的個數5,而後取模,映射到環形數組裏的數據元素,假如當前時間是1 000 000 002s,那麼對應當前時間的環形數組裏的第3個元素,下標爲2。

此時的數組元素的數據如圖1-23所示。


圖1-23

在圖1-23中,當前時間爲1 000 000 002s,對應的計數器在第3個元素,下標爲2,當前請求是在這個時間週期內的第1個訪問請求,程序首先須要對後一個元素即第4個元素,也就是下標爲3的元素清零;在1 000 000 002s內,任何一個請求若是發現下標爲3的元素不爲0,則都會將原子變量3清零,並記錄清零的時間。

這時程序能夠對第3個元素即下標爲2的元素,進行累加並判斷是否達到閾值,若是達到閾值,則拒絕請求,不然請求經過;同時,打印本次及以前3秒的數據訪問量,打印結果以下。

當前:1次,前1s:302次,前2s:201次,前3s:518次

然而,若是當前秒一直沒有請求量,下一秒的計數器始終不能清零,則下一秒的請求到達後要首先清零再使用,並更新清零時間。

在下一秒的請求到達後,若檢查到當前秒對應的原子變量計數器不爲0,並且最後的清零時間不是上一秒,則先對當前秒的計數器清零,再進行累加操做,這避免發生上一秒無請求的場景,或者上一秒的請求因爲線程調度延遲而沒有清零下一秒的場景,後面這種場景發生的機率較小。

另一種實現計數器的簡單方法是單獨啓動一個線程,每隔必定的時間間隔執行對下一秒的原子變量計數器清零操做,這個時間間隔必須小於計數時間間隔。

2)令牌筒

令牌筒是一個流行的實現限流的技術方案,它經過一個線程在單位時間內生產固定數量的令牌,而後把令牌放入隊列,每次請求調用須要從桶中拿取一個令牌,拿到令牌後纔有資格執行請求調用,不然只能等待拿到令牌再執行,或者直接丟棄。

令牌筒的結構如圖1-24所示。


圖1-24

3)信號量

限流相似於生活中的漏洞,不管倒入多少油,下面有漏管的流量是有限的,實際上咱們在應用層使用的信號量也能夠實現限流。
使用信號量的示例以下:

public class SemaphoreExample {
    private ExecutorService exec = Executors.newCachedThreadPool();
    public static void main(String[] args) {
        final Semaphore sem = new Semaphore(5);
        for (int index = 0; index < 20; index++) {
            Runnable run = new Runnable() {
                public void run() {
                    try {
                        // 得到許可
                        sem.acquire();
                        // 同時只有5個請求能夠到達這裏
Thread.sleep((long) (Math.random()));
                        // 釋放許可
                        sem.release();

                        System.out.println("剩餘許可:" + sem.availablePermits());
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            };
            exec.execute(run);
        }
        exec.shutdown();
}
}複製代碼

4. 失效轉移模式

若微服務架構中發生了熔斷和限流,則該如何處理被拒絕的請求呢?解決這個問題的模式叫做失效轉移模式,一般分爲下面幾種。

  • 採用快速失敗的策略,直接返回使用方錯誤,讓使用方知道發生了問題並自行決定後續處理。
  • 是否有備份服務,若是有備份服務,則迅速切換到備份服務。
  • 失敗的服務有多是某臺機器有問題,而不是全部機器有問題,例如OOM問題,在這種狀況下適合使用failover策略,採用重試的方法來解決,可是這種方法要求服務提供者的服務實現了冪等性。
  • 說到這裏順便給你們推薦一個Java方面的交流羣650385180,裏面會分享一些資深架構師錄製的視頻錄像:有Spring,MyBatis,Netty源碼分析,高併發、高性能、分佈式、微服務架構的原理,JVM性能優化這些成爲架構師必備的知識體系。還能領取免費的學習資源,如下的知識體系圖也是在羣裏獲取。相信對於已經工做和遇到技術瓶頸的朋友,在這個羣裏會有你須要的內容。


微服務的粒度

在服務化系統或者微服務架構中,咱們如何拆分服務纔是最合理的?服務拆分到什麼樣的粒度最合適?

按照微服務的初衷,服務要按照業務的功能進行拆分,直到每一個服務的功能和職責單一,甚至不可再拆分爲止,以致於每一個服務都能獨立部署,擴容和縮容方便,可以有效地提升利用率。拆得越細,服務的耦合度越小,內聚性越好,越適合敏捷發佈和上線。

然而,拆得太細會致使系統的服務數量較多,相互依賴的關係較複雜,更重要的是根據康威定律,團隊要響應系統的架構,每一個微服務都要有相應的獨立、自治的團隊來維護,這也是一個不切實際的想法。

所以,這裏倡導對微服務的拆分適可而止,原則是拆分到可讓使用方自由地編排底層的子服務來得到相應的組合服務便可,同時要考慮團隊的建設及人員的數量和分配等。

有的公司把每一個接口包裝成一個工程,或者把每一次JDBC調用包裝成一個工程,而後號稱是「微服務」,最後有成百上千的微服務項目,這是不合理的。固然,有的公司把一套接口完成的一個粗粒度的流程耦合在一個項目中,致使上層服務想要使用這套接口中某個單獨的服務時,因爲這個服務與其餘邏輯耦合在一塊兒,因此須要在流程中作定製化才能實現使用方使用部分服務的需求,這也是不合理的,緣由是服務粒度太粗。

總之,拆分的粒度太細和太粗都是不合理的,根據業務須要,可以知足上層服務對底層服務自由編排並得到更多的業務功能便可,並須要適合團隊的建設和佈局。

相關文章
相關標籤/搜索