本文探討以下內容:編程
要理解Entity和VO,須要先理解兩個概念:「狀態」和「標識」!咱們先來聊聊「狀態」!網絡
你們確定都在淘寶買過東西吧!在淘寶購買商品後,會有一個訂單,記錄了你購買的商品信息、價格、店鋪信息、還有一個特別重要的信息,就是訂單狀態。經過這個訂單狀態,咱們能夠知道咱們的購物流程如今進行到哪一步了。若是你猶豫了好久才下定決心購買了一件心儀已久的商品,你是否是很在乎訂單狀態?時不時要刷新一下頁面,看看訂單狀態是否顯示已送達了?架構
開發過系統的都知道,通常訂單狀態都是使用一個字段來表示的,好比status,不一樣的狀態就是給status賦不一樣的值。可是這個status就是「訂單狀態」嗎?難道狀態就是一個字段?!併發
Order{ product location seller buyer status ... }
你有沒有想過,當咱們說「狀態」的時候,咱們實際上指的是什麼?編程語言
咱們在不少場景下會用到「狀態」這個詞,好比:分佈式
以「你今天狀態不錯」這句爲例,若是狀態就是一個字段!那麼,「你今天狀態不錯」就是status=1?!「你今天狀態不行」就是status=0?!很明顯,這不合理!動畫
若是「狀態」不是簡單的一個字段的話,那麼「狀態」究竟是什麼呢?設計
其實在架構風格:你真的懂REST嗎?已經提過了!文中對REST的解釋,有這麼一句:一個由網頁組成的網絡(一個虛擬狀態機),用戶經過選擇連接在應用中前進(狀態遷移),致使下一個頁面(應用的下一個狀態的表述)被轉移給用戶,而且呈現給他們,以便他們來使用。指針
結合上面的幾個場景,你有沒有發現,「狀態」實際上表示的是「目標對象在當前時刻所呈現出的內容」!在軟件系統中經過一個字段來表示狀態只是一種簡化手段!對象
如無特殊說明,下面所提到的「狀態」指的是「目標對象在當前時刻所呈現出的內容」,而不是指狀態字段
既然「狀態」表示的是「當前時刻所呈現出的內容」!那麼說明了「狀態」是個快照/瞬態!也就是說,「目標對象」有多個「狀態」,「當前狀態」只是「目標對象」衆多「狀態」中的一個!
你們應該玩過定格動畫吧?就像下面這樣(下圖截自《大偵探福爾摩斯2:詭影遊戲》):
圖中的小冊子就是「目標對象」,冊子的每一頁就是「狀態」,當前展現出來的那一頁就是「當前狀態」!
在理解了什麼是「狀態」之後,咱們就能夠來初步區分Entity和VO了:
如今,問題又來了,對於VO來講,由於「狀態」是不可變的,咱們就能夠用其「狀態」來表示VO!可是對於Entity來講,由於有多個「狀態」,且「狀態」是可變的,那咱們如何來表示呢?以上面的Order爲例,假設同一個買家在同一個賣家那裏買了兩個一樣的商品,那兩個訂單裏的信息都是同樣的,可是它是兩個不一樣的訂單,咱們如何區分這兩個訂單呢?
如今就輪到下一個主角登場了:「標識」!
說到「標識」,咱們最早想到的是編程語言中的「引用」或「指針」!好比下面的代碼:
Order orderA = new Order("productA",...); Order orderB = new Order("productA",...); orderA.productName = "productB";
這解決了「區分相同狀態的不一樣Entity」的問題,可是沒有解決Entity有多個狀態的問題。由於「標識」指向的是目標對象的當前狀態。並且,不少編程語言中有個很大的問題,就是不區分「標識」和「狀態」!什麼意思呢?
假設咱們在看一部電影,當咱們開始觀看時,就是這部電影生命週期的開始,觀看結束就是這部電影生命週期的結束,在這段時間裏,電影的畫面(狀態)一幀幀的呈如今咱們面前,咱們能夠經過播放、快進、後退、暫停改變電影的狀態,每一個狀態都是相互獨立的,相似這樣:
隨着時間的改變,咱們能獲取到電影的不一樣狀態,每一個狀態是相互獨立的。可是實際上咱們的代碼邏輯像下面這樣:
var movie1 = new Movie(); movie1.setCurrentFrame("第三幀"); var currentMovie = movie1 movie1.setCurrentFrame("第四幀"); currentMovie // 仍是第三幀嗎?
電影播放到第三幀,咱們用一個變量currentMovie保存了電影的當前狀態(第三幀),可是後面電影播放第四幀了,currentMovie也就變成了第四幀的狀態了。
語言中的這種「標識」(我稱爲「隱式標識」)還有另一個問題,就是沒法跨系統。好比,在分佈式系統中,須要保證兩個系統中的對象是同一個對象,這種「隱式標識」是作不到的。
因此「隱式標識」並不能知足咱們的需求。咱們須要「顯示標識」,「顯示標識」在現實中很常見:
在上面購物的列子中,就至關於給Order一個惟一標識,好比一個惟一的訂單號:
Order{ orderNo // 顯示標識 product location seller buyer status ... }
給定訂單號之後,不管訂單的狀態如何變化,只要訂單號不變,那麼它就是同一個訂單。
因此,「標識」是另外一個區分Entity和VO的關鍵點:
注意標識並不必定只是一個字段,多是多個字段的組合,這須要根據不一樣的業務邏輯來肯定。好比在一個學校系統裏,能夠經過學年+班級+學號來標識一個學生。
理解了標識和狀態,咱們就能夠來定義Entity和VO了:
如今咱們知道了什麼是Entity,什麼是VO,那麼咱們如何在系統中識別哪些對象是Entity,哪些對象又是VO呢?
一個對象是表示成Entity仍是VO,取決於系統的關注點。
咱們還以淘寶購物爲例,假設你在某家店鋪買了個商品,質量很好。過了一段時間後,你想再買一個,可是你記不得是哪家店了,因而你從已完成的訂單列表中點擊商品想進去再次購買。可是你點進去後發現,商品下架了。
這是由於「商品」在「訂單系統」中是個VO,而在「商品管理系統」中是Entity!其實很好理解:
在「商品管理系統」中,商品能夠這樣表示:
Product { id // 商品標識 name desc status ... }
而在「訂單系統」中,訂單是個Entity,商品是個VO,能夠這麼表示:
Order{ orderNo // 訂單標識 product:Product status ... } Product { id // 這裏不是標識,只是狀態 name desc status ... }
注意這裏的id並非標識,這裏的id實際上退化成了狀態的一部分,保留這個id是爲了和「商品管理系統」進行交互,經過id從商品管理系統中查詢商品。固然還有其它方式,例如保存「商品管理系統」中該商品的歷史URL。
本文從對「狀態」和「標識」的理解開始,一步步來解釋什麼是Entity和VO,以及如何在系統中識別Entity和VO。後面將進一步討論Entity與VO的關係,以及與其它組件的關係,例如DTO,Service,Resporitory,DAO等