DDD早於微服務「出道」十年,這兩個「忘年交」的軟件設計哲學是如何相愛相殺的?前端
微服務如今能夠說是軟件研發領域無人不提的話題,然而業界流行的對比多數都是所謂的Monolithic(單體應用),而大量的系統在十幾年前都已是以SOA(面向服務架構)爲基礎的分佈式系統了,那麼微服務做爲新的架構標準與SOA有什麼差別點呢?其本質區別在於設計原理,微服務是去中心化設計,SOA是「集成」造成中心設計;
程序員
另外,筆者認爲如下幾點並非微服務和SOA的區別點:數據庫
CI/CD:持續集成、持續部署自己與敏捷、DevOps是交織在一塊兒的,CI\CD更傾向於軟件工程的領域,與微服務無關;編程
通信協議:微服務的推薦通信協議是RESTful,而傳統的SOA是SOAP。不過基於輕量級的RPC框架Dubbo、Thrift、gRPC來實現微服務也不少;在Spring Cloud中也有Feign框架將標準RESTful轉爲代碼的API這種仿RPC的行爲,這些通信協議不是區分微服務架構和SOA架構的核心差異;設計模式
固然,軟件工程(DevOps)、基礎設施(容器化)、軟件開發模式(敏捷開發)的變革有利的推動了微服務架構的大行其道。而微服務架構是一種架構風格、架構理念,其中的「微」更體現了它的精髓在切分。在實際微服務的落地過程當中證實,若是切分是錯誤的,你得不到微服務承諾的「低耦合、自治、易維護」之類的優點,而且還會比單體架構擁有更多的麻煩。那麼如何切分呢?其實並非一些新的方法論,而是都提出不少年的架構設計方法,也稱它們爲微服務設計基礎或架構模型:領域驅動設計和立方體模型。數據結構
2004年,Eric Evans 發表了Domain Driven Design(領域驅動設計,DDD)。領域驅動設計已經問世十幾年,從Eric Evans出版的著做「領域驅動設計」一書中對領域驅動作了開創性的理論闡述,在軟件設計領域中,DDD能夠稱得上是步入暮年時期了。遺憾的是,國外軟件圈享有盛譽並行之有效的設計方法學,國內大多數的技術人員卻並不瞭解,也不曾運用到項目實踐中。直到行業內吹起微服務的熱風,人們彷佛才從新發現了領域驅動設計的價值,並非微服務拯救了領域驅動設計,是由於領域驅動設計一直在頑強的成長,其設計開放的設計方法體系,雖然歷來未曾在國內大行其道,但卻發揮着巨大的價值。表面上看確實是由於微服務,領域驅動設計才又開始出如今大衆視野裏。架構
固然,領域驅動設計並不是「銀彈」,不是能解決全部疑難雜症的「靈丹妙藥」,學習並應用它的意義在於:框架
領域驅動設計與微服務架構天生匹配,不管是在新項目中設計微服務架構,仍是將系統從單體架構演進到微服務設計,均可以遵循領域驅動設計的架構原則。
固然,領域驅動能給咱們帶來不少收穫,但若是你是屬於如下幾種狀況的某種,那麼你確實不須要學習領域驅動設計了:dom
若是你負責的軟件系統並不複雜,二三人即可輕鬆維護數據庫設計
一個軟件系統的誕生,必定是爲了解決咱們遇到的某個問題。好比一家企業的一直採用線下銷售產品,耗費大量的財力和物力,但願能夠在線上銷售本身的產品,用於實如今線銷售銷售產品的目的,那麼就誕生了一個電商系統。一般最初設立的目標或要解決的問題就是一個軟件項目的出發點,明確咱們要作什麼。好比一個電商、一個論壇、一個支付平臺等。
下文將從領域、問題域、領域模型、設計、驅動這幾個詞語的含義和聯繫的角度去闡述DDD是如何融入到軟件開發的。要理解什麼是領域驅動設計,首先要理解什麼是領域,什麼是設計,什麼是驅動,什麼驅動什麼。
領域是與某個特定問題相關的知識和行爲。好比支付平臺就屬於特定的領域,只要是這個領域,都會有帳戶、會記、收款、付款、風控等核心環節。因此,同一個領域的系統都具備相同的核心業務,他們要解決的問題的本質是一致的。一個領域本質上能夠理解爲就是一個問題域,只要是同一個領域,那問題域就相同。因此,只要咱們肯定了系統所屬的領域,那這個系統的核心業務,即要解決的關鍵問題、問題的範圍邊界就基本肯定了。
在平常開發中,咱們一般會將一個大型的軟件系統拆分紅若干個子系統。這種劃分有多是基於架構方面的考慮,也有多是基於基礎設施的。在DDD中,咱們對系統的劃分是基於領域(基於業務)的。好比上文提到支付平臺是一個領域,而帳戶、會記、收款、付款等則爲子領域。一個領域由衆多子領域彙集而造成。
固然,問題隨之而來:
DDD中,有標準方法解決上述問題,就是限界上下文(Bounded Context)和上下文映射圖。在一個領域/子域中,咱們會建立一個概念上的領域邊界,在這個邊界中,任何領域對象都只表示特定於該邊界內部的確切含義。這樣的邊界便稱爲限界上下文。限界上下文和領域具備一對一的關係。從物理層面講,一個限界上下文最終能夠是一個Jar/War文件,甚至能夠是一個Package中的全部對象。可是,技術自己並非用來界分限界上下文。
上圖引自《實現領域驅動設計》。一般狀況下,一個領域有且只有一個核心問題,咱們稱之爲該領域的「核心域」。在覈心域、通用子域、支撐子域梳理的同時,會定義出子域中的「限界上下文」及其關係,用它來闡述子域之間的關係。界限上下文能夠簡單理解成一個子系統或組件模塊。
DDD中的設計主要指領域模型的設計。DDD是一種基於模型驅動開發的軟件開發思想,強調領域模型是整個系統的核心,領域模型也是整個平臺的核心價值。每個領域都有一個對應的領域模型,領域模型可以很好的解決負責的業務問題。因此領域模型的設計和架構設計同等重要。
DDD中,老是以領域爲邊界,分析領域中的核心問題(核心關注點)。而後設計對應的領域模型,經過領域模型驅動代碼的實現。而數據庫設計、持久化技術這些都不是DDD的核心,屬於外圍的東西。與數據庫驅動開發的思路造成對比,驅動中須要記住兩個原則:
領域模型驅動代碼實現
領域驅動設計的最大價值是讓咱們告別從面向過程式的思想(天馬星空,想到哪寫到哪)轉化爲基於系統化的模型驅動思惟。咱們腦補一下軟件開發中的常規心路歷程:
七、重複執行3-6步......
Eric Evans在《領域驅動設計-軟件核心複雜性應對之道》這本書中提出了傳統的四層架構模式,在後來演進過程當中出現了五層架構和六層架構,,以下圖所示:
隨着後續的演進,出現了一種改進分層架構的方法,即Robert C. Martin提出的依賴倒置原則(Dependency Inversion Principle,DIP)。它經過改變不一樣層之間的依賴關係達到改進目的。
根據該原則的定義,DDD分層架構中的低層組件應該依賴於高層組件提供的接口,即不管高層仍是低層都依賴於抽象,整個分層架構好像被推平了,再向其中加入了一些對稱性,就出現了一種具備對稱性特徵的六邊形架構風格。六邊形架構是Alistair Cockburn在2005年提出的,其本質是倡導不一樣的客戶經過「平等」的方式與系統交互,經過不斷的擴展適配器轉化成系統API所理解的參數來達到每種特定的輸出,而每種特定的輸出都有適配器完成相應的轉化功能。
讀完上面的兩種分層架構方式,可能不少人會有疑問,這些是什麼?爲何我以前一直都沒聽到過這種分法?確實是這樣,DDD和麪向對象、設計模式等等理論有千絲萬縷的聯繫,若是不熟悉OOA、OOD,那麼DDD可能也會理解不了。由於咱們大部分從開發生涯開始之初接觸的都是「Action層、Service層、Dao層、DB層」這樣的MVC分層理論。而且在21中設計模式中,「行爲型」的設計模式,咱們幾乎沒有什麼機會使用,致使這些問題的緣由是J2EE經典分層的開發方式是「貧血模型」。
Martin Fowler(對,就是提出微服務的那位大牛)曾經提出了兩種開發方式,即:
以「充血模型」爲基礎的「領域驅動」的開發方式
貧血模型是指對象只用於在各層之間傳輸數據使用,只有數據字段和Get/Set方法,沒有邏輯在對象中。而「事務腳本」能夠理解爲業務是由一條條增刪改查的SQL組織而成,是面向過程的編程。
充血模型是面向對象設計的本質,一個對象是擁有狀態和行爲的。將大多數業務邏輯和持久化放在領域對象中,業務邏輯只是完成對業務邏輯的封裝、事務、權限、校驗等的處理。
舉例,用戶管理模塊大概是這樣的兩種實現:
// 貧血模型下的實現
public class User{
private Integer id; private String name; ... // 省略get/set方法
}
public class UserManager{
public void save(User user){
// 持久化操做.... }
}
// 保存用戶的操做多是這樣
userManager.save(user);
// 充血模型下的實現
public class User{
private Integer id;
private String name;
...
// 省略get/set方法
public void save(){ // 持久化操做.... }
}
// 保存用戶的操做多是這樣
user.save();
Martin Fowler定義的「貧血模型」是反模式,面對簡單的小系統用事務腳本方式開發沒問題;稍微大一些的系統使用事務腳本方式會擴大維護成本,業務邏輯、各類狀態散佈在大量的函數中,哪怕就是要用戶對象中增長一個字段,可能都會涉及到幾個類的調整......
但願領域對象可以準確地表達出業務意圖,可是多數時候,咱們所看到的倒是充滿getter和setter的領域對象,此時的領域對象已經不是領域對象了,反模式的貧血對象了。其實在貧血模型和充血模型模型以外,還有失血模型和脹血模型,但後者兩個基本是實際開發中不會去使用,由於走的是兩個極端。
本文宏觀角度介紹了領域驅動設計,那麼微服務和DDD是什麼關係呢?其實在2015年的一次演講中,DDD的提出者Eric Evans表達了對微服務技術的熱愛與支持,認爲微服務是讓DDD落地的好工具。由於DDD和微服務其本質是下降軟件項目的複雜性,而DDD是一種設計理念/設計方法,DDD須要有強制性的原則作保障,不然不一樣的領域對象終究會混在一塊兒。而微服務自己的一些限制,以及你們都能理解微服務的實施前提和首要條件,會在實現上給DDD增長了一些原則限制。DDD和微服務的不必定要同時使用落地,可是若是將DDD和微服務(兩個相差十歲的軟件設計方法)結合一塊兒,那麼Martin Fowler和Eric Evans兩位佈道師是會很贊同的。