無需If-Else語句,狀態模式即可編寫干凈可維護的代碼
本文轉載自公眾號“讀芯術”(ID:AI_Discovery)。
If-Else語句我們再熟悉不過了,你可能看過無數的相關教程,讀過很多編程書籍,來提升將If-Else用作de facto分支技術的方法,甚至可能你的默認模式就是使用If-Else。
可以結束這一切了。在任何新的條件要求下,使用If-Else都會增加復雜性。用狀態對象來代替If-Else吧。應用狀態模式,你只需使用專用狀態對象而無需使用If-Else語句來更改對象行為。
像下面這樣的代碼時代已經一去不復返了:
觸發PTSD警告
你肯定曾經寫過更復雜的分支,我也曾是這么做的。上面的分支邏輯甚至還不是很復雜,但如果嘗試添加新條件,事情肯定會越弄越糟。
另外,如果你認為創建新類而非簡單地使用分支語句聽起來很煩人,那么請一定等到實際使用的時候,你會發現這種方法簡潔而優雅。除了“ D”部分之外,它還將使你的代碼庫更加扎實。
那么如何避免混亂的分支代碼呢?
我們將研究如何在生產就緒代碼中替換If-Else分支。這是一個虛構的例子,但是方法與我在大型客戶的代碼庫中使用的方法相同。
創建一個非常簡單的Booking類,其中包含一些狀態。它還有兩個公共方法:Accept()and Cancel().。我盡力繪制了一個圖表,顯示了預訂可能處于的不同狀態。
代碼中重構分支邏輯的過程一共有三步:
- 創建一個抽象基類
- 將每個狀態實現為繼承自基本狀態的單獨類
- 讓Booking`類具有私有或內部方法,該方法以狀態基類為參數
演示時間到!
首先,需要一個將繼承所有狀態的基類。
請注意,該基類還具有兩種方法,Accept和Cancel——盡管此處將它們標記為內部。此外,基態具有“特殊” EnterState(Booking booking)方法。每當將新狀態分配給預訂對象時,就會調用此方法。
其次,為要代表的每個狀態創建單獨的類。
注意,如上圖所示,每個類如何表示一個狀態。此外,ExpiredState和CancelledState不會將預訂轉換為新狀態。這兩類在本質上與Null Object Pattern非常相似。
最后,預訂類型本身。
看到預訂類型如何簡單地將Accept和Cancel的實現委派給其狀態對象了嗎?這樣做可以使我們刪除許多條件邏輯,并使每個狀態僅關注對自己重要的事情,當前狀態也有機會將預訂過渡到新狀態。
Q&A
1. 如何處理新的條件功能?
如果說通常需要使用某些條件檢查來實現新功能,那么現在只需創建一個新的狀態類即可,你將不再需要費力地處理if-else語句了。
2. 如何將狀態對象保留在數據庫中?
不,你不想。
當將對象保存到例如SQL或NoSQL數據庫時,狀態對象并不重要,這時只有了解對象的狀態及其應如何映射到列是重要的。你可以將狀態映射到友好類型名稱,枚舉或整數。只要你愿意,總有某種方法可以將保存的值轉換回狀態對象。
3. 但是你還在使用If’s嗎?
它們是必不可少的,特別是當用作保護子句時。但If-Else組合是造成可維護性難題的根本原因。
圖源:unsplash
復雜性不是源于你擁有的類的數量,而是源于這些類承擔的功能。擁有許多專門的類將使你的代碼庫更具可讀性與可維護性,并且從總體上來說,也更易于使用。
適時地和If-Else語句說再見吧!