元件如何正確歸類 ? | 元件內聚性三大原則 : REP、CCP、CRP

 


元件如何正確歸類 ? | 元件內聚性三大原則 : REP、CCP、CRP

適當時機 x 適當分類

大綱

  • 全新的專案
  • 高內聚就能低耦合
  • 內聚性三大原則
    • REP , 再使用性 - 發佈等價原則
    • CCP , 共同封閉原則
    • CRP , 共同重複使用原則
  • 內聚性張力圖
  • 適當的時機,適當的分類

全新的專案

全新的專案,從無到有的開發

過往的經驗

打算把很多重複的功能,都聚集成 一個獨立的元件

類別分類

要將哪個類別歸類到哪個元件,是個重大的決定!

但大部分的時候,卻都是憑感覺分類。

內聚性三大原則

就是在告訴你 :

分類時,可以依據類別的什麼特性,來歸類成同一個元件。

除此之外 :

它甚至告訴你,現在就將類別,抽成共用元件,並不是恰當的「時機點」。


高內聚就能低耦合

延續元件耦合性的部分


在上一篇章中,已經知道:

元件之間建立關係,不應該是那麼的「隨心所欲」

高內聚,低耦合

高內聚在前,除了類別要先內聚成大顆粒的元件,才能互相依賴以外。從元件內部來看,如果都是相似且恰當的類別, 那麼元件在要使用特定功能時,要依賴的元件,就不會是分佈於程式各處,而會相對集中。

元件只會依賴必要的元件,整體系統的參照數量,就會降低。

同時也達成「低耦合」的目標

內聚性三大原則

REP 再使用性 - 發佈等價原則

再使用性的細緻度就是發佈的細緻度

CCP 共同封閉原則

將那些會因著相同理由、再相同時間發生變化的類別,收集到相同的元件之中。
將那些在不同時間、因著不同理由發生變化的類別,分割到不同的元件之中。

CRP 共同重複使用原則

不要強迫元件的使用者依賴他們不需要使用的東西


REP , 再使用性 - 發佈等價原則

The Reuse/Release Equivalence Principle

再使用性的細緻度就是發佈的細緻度

這個原則是再說明,真正可以「再使用」的「元件」,必須是像發佈應用程式一樣;

擁有獨立的版本號與說明文件能夠「追蹤」與「紀錄」每一個時期,程式的變化,才是符合「再使用」的元件特性。

也就是

再使用的「元件」,必須是已經「發佈」過的元件

為什麼 ?

這麼做的目的,是不希望當要使用現成的程式功能時,只是純粹的程式碼「複製」、「貼上」

後續如果那段程式被更動了,追蹤與管理都會是一個大問題。


再使用性

更嚴謹的做法,就是將要「重用」的程式,獨立成一個可以任意部屬的元件。 在其他應用程式中,只需要知道外部的接口,而不需要知道原始的程式碼。

能夠方便的使用,也是「再使用性」的特徵

CCP , 共同封閉原則

The Common Closure Principle

將那些會因著相同理由、再相同時間發生變化的類別,收集到相同的元件之中。
將那些在不同時間、因著不同理由發生變化的類別,分割到不同的元件之中。

延續「共同封閉原則」

上一個 REP 原則,說明 要「再使用」的元件,必須要像應用程式一樣發佈

因此,元件內部的類別組成,就必須要能夠實際的解決:

  • 特定的主題
  • 特殊的目的
也就是「內聚性」必須要強

但該原則並沒有具體的說明,怎樣才算是「內聚性」強

「CCP 共同封閉原則」與 「CRP 共同重複使用原則」就是用來補充 REP 原則,未具體說明的部分。


從描述上來看,可以很明顯地知道。分類的依據就是根據「變化」的可能性進行分類


為什麼 ?

因為軟體雖然是可變的,但還是希望「最少改動,最大收益」。

如果程式需要被更改,那麼當然希望更改的地方,都集中在同一個元件,而不是分散開來,到處都是。


CRP , 共同重複使用原則

The Common Reuse Principle

不要強迫元件的使用者依賴他們不需要使用的東西

延續「CCP 共同封閉原則」

使用「變化」的可能性,進行分類後。

這個原則,則是要將分類好的元件內部,在做一個更細緻的分類。


為什麼 ?

因為一個元件中,使用者有可能只用了這個元件的很小一部分,其他的部分並不是使用者所需要的。


但依賴的關係仍然是建立,這個元件其他部分更動了,還是必須要重新編譯、驗證、佈署。

如果有更精簡的元件可以使用,就沒有理由非得要選擇你的元件


所以,為了更強的內聚性,就是以使用者的角度,將元件進行分割

目標

元件內的類別,都是使用者剛好要使用到的部分 (元件不要依賴不需要的東西)

內聚性張力圖

依據上述的三個原則,可以發現每個原則,所關注的重點都不太一樣 :

REP 再使用性 - 發佈等價原則

目標: 讓元件更好的被使用

關注的是「重用性」

CCP 共同封閉原則

目標: 讓元件更好的被維護

關注的是「發展性」

CRP 共同重複使用原則

目標: 讓元件更精準的定位

關注的是「精確性」

這三個原則,彼此之間會互相拉扯,太多的關注某些原則,就會導致另外一個原則無法被確切的符合。


這種情況,可以描述成一個張力圖:

當開發者只關注其中的兩個頂點,而放棄另外一個頂點的原則時,程式必須要付出的代價。


只關注 CCP、CRP 原則,不在乎 REP 再使用性 - 發佈等價原則

元件就會很難被重用

只關注 REP、CRP 原則,不在乎 CCP 共同封閉原則

修改時就會有太多的元件要被更動

只關注 REP、CCP 原則,不在乎 CRP 共同重複使用原則

造成太多不必要的版本發佈

第四維度

因此在軟體設計中,還有一個必須要考慮的因素:

時間維度

對於大多數的應用程式,由於想要 :

  • 增加「更多的功能」
  • 滿足「更多的需求」

「可發展性」的重要程度通常會大於「可重用性」

也就是說,在專案開發初期:

CCP 原則的重要性會大於 REP原則(可發展比重複使用更重要)

所以這時,第一個應該要思考的是

如何恰當的「分類」,而不是思考未來該如何「重用」。


聚合原則重要性演變

重用的部分,必須加入了 CRP 原則,更精簡後,依循 REP 原則正式發佈:

此時,你的元件才會真正意義上的,具有「高內聚」的特性。


適當的時機,適當的分類

回頭來看 DDD 領域驅動設計

四個分層,其實就是 CCP 原則的概略呈現。


而在每一層中,我通常的做法,都會在更細緻的依據業務情境,進行分類:

Domain 層

每個業務情境都會有各自的 Service Entity 與 vo


Application 層

每個業務流程也都會有各自的 Service 與 DTO

這種細緻化的方式,也是 CCP 原則 與 CRP 原則的綜合應用。


REP 原則,具體的呈現

就像是我的 :

JavaProjUtil

iOSProjUtil

各種 Util 結尾的專案,可以獨立的發佈版本與提供說明文件。


動態的軟體,動態的原則

軟體是一個動態的東西,設計軟體元件內部的內聚性原則也是動態的。

身為優秀的軟體工程師,可能就會意識到,這個就是:

「軟體的本質」 與 「軟體的特性」

掌握了這些原則,你就能:

在適當的時機點,出現適當的東西。

軟體的天堂之境:「最少改動,最大收益」,已經出現在不遠處。


語錄

工程師紀錄

有想法的人很多,有作法的人很少

            -- 我以前的組長

參考資料

留言

熱門文章

Markdown 語法大全,範例模板

【如何寫乾淨的程式碼 ? 】程式設計 代碼風格 指南 | 基礎 + 9 個進階概念

【 git 基礎教程 #1】什麼是 git ? | Sourcetree 介紹 與 入門基礎操作教學

【什麼是 git flow ?】 5 項分支全詳解 | Sourcetree 實戰演練