單元測試引出重構(gòu)貧血模型的嘗試

模型的四種血量分類:

失血模型:property都是 { get; set; }独旷,沒有任何業(yè)務(wù)邏輯。
貧血模型:比如property的get內(nèi)部有業(yè)務(wù)邏輯,業(yè)務(wù)邏輯由模型內(nèi)部數(shù)據(jù)即可計算得出糜值,不依賴于外部。
充血模型:比如property的get內(nèi)部有業(yè)務(wù)邏輯踪央,業(yè)務(wù)邏輯依賴于外部接口臀玄,比如持久化接口。
脹血模型:模型內(nèi)部直接實現(xiàn)所有依賴畅蹂,接口都不需要了健无。

模型和上下文場景

class Order
{
    DateTime CreatedTime { get; }

    List<Item> Items { get; }

    bool IsAvailable 
    { 
        get 
        { 
            return Items.Count > 0; 
        }
    }
}
  1. Order是貧血模型,沒有外部接口依賴液斜。
  2. Order的property IsAvailable是我們關(guān)注和重構(gòu)的核心累贤。
  3. Order的物理結(jié)構(gòu)不會變化,業(yè)務(wù)復雜性只在IsAvailable的邏輯規(guī)則上增長少漆。
  4. Order由OrderManage的GetOrder方法提供實例臼膏。
  5. IsAvailable被廣泛使用,如下所示各種consumer示损。
class OrderManager : IOrderManage
{
    public Order GetOrder() {...}
}
class OrderConsumerA
{
    IOrderManage OrderManager

    public void ConsumerABusiness()
    {
        var isOrderAvailable = OrderManager.GetOrder().IsAvailable;
        ...
    }
}

class OrderConsumerB
{
    IOrderManage OrderManager

    public void ConsumerBBusiness()
    {
        var isOrderAvailable = OrderManager.GetOrder().IsAvailable;
        ...
    }
}

class OrderConsumerC
{
    IOrderManage OrderManager

    public void ConsumerCBusiness()
    {
        var isOrderAvailable = OrderManager.GetOrder().IsAvailable;
        ...
    }
}

單元測試代碼

使用單元測試框架Mock

class OrderConsumerATests
{
    Mock<IOrderManage> mockOrderManager

    public void ConsumerABusiness()
    {
        mockOrderManager.Setup(x => x.GetOrder()).Returns(new Order { Items = new List<Item> { new Item() }});
        ...
    }
}

Order是Model渗磅,沒有基于抽象定義。無法使用Mock框架構(gòu)造一個fake的IsAvailable實現(xiàn)去直接返回我們testcase期望的結(jié)果。因此需要在每個testcase內(nèi)構(gòu)建一個Order實例始鱼。
當業(yè)務(wù)邏輯簡單且模型結(jié)構(gòu)簡單時仔掸,那么在各個consumer單元測試中重復構(gòu)造整個對象并不費力,我們不覺得痛医清,還可以接受的起暮。

但是,IsAvailabled的業(yè)務(wù)邏輯變復雜時

class Order
{
    bool IsAvailable 
    { 
        get 
        { 
            return Items.Count > 0 && (CreatedTime.AddDays(7) >= DateTime.Now || (Items.All(item => !item.HasInventory) && Items.Sum(item => item.Price) < MaximumPrice)) ; 
        }
    }
}

復雜度上升会烙,導致構(gòu)造適合每個testcase的Order對象變得復雜负懦,IsAvailable內(nèi)部邏輯在每個testcase中都會被執(zhí)行一次。維護單元測試就變得越來越困難柏腻,這是痛點纸厉。

想法

  • 單元測試只關(guān)心單元內(nèi)部的邏輯和實現(xiàn),不要有類似于集成測試的單元測試
  • 單元測試準備數(shù)據(jù)的過程不要太繁瑣葫盼,寫單元測試不要被準備數(shù)據(jù)的過程所阻礙
  • 給IsAvailable一個virtual關(guān)鍵字就可以使用Mock<Order>對象解決問題残腌,但是感覺只是為了解決問題而改動,我們需要更好的方案

現(xiàn)在贫导,如果是你負責在Order的IsAvailable上再增加業(yè)務(wù)邏輯抛猫,面對越來越難以維護的單元測試,你會怎么重構(gòu)代碼孩灯?你會怎樣繼續(xù)寫單元測試去覆蓋新增加的業(yè)務(wù)邏輯闺金?現(xiàn)在, 不妨停下來思考一下先.

OK峰档,已有的想法是败匹,

1. 縱向-提取抽象到父類

給Order 一個父類OrderBase,對應(yīng)的abstract/virtual 關(guān)鍵字體現(xiàn)抽象讥巡。接下來只要Mock的setup就好了掀亩,不必多說。

class OrderBase
{
  virtual bool IsAvailable(Order order);
}

Order上方的父類就是對業(yè)務(wù)邏輯抽象結(jié)構(gòu)的體現(xiàn)欢顷。理論上講與virtual IsAvailable性質(zhì)一樣槽棍,但是在model整體上做到了面向?qū)ο螅嫦虺橄蟆?br> 這種抽象是對model本身結(jié)構(gòu)的抽象抬驴。

2. 橫向-提取邏輯到接口

另外一條思路就是提取新的接口IOrderService炼七,業(yè)務(wù)邏輯向接口實現(xiàn)中遷移。所以這種方法會讓模型流血布持,有反內(nèi)聚的嫌疑豌拙,但是是否適用得具體問題具體分析。

interface IOrderService
{
   bool IsAvailable(Order order);
}

然后题暖,Consumer引入

class OrderConsumerA
{
    IOrderManage OrderManager

    IOrderService OrderService

    public void ConsumerABusiness()
    {
        var isOrderAvailable = OrderService.IsAvailable(OrderManager.GetOrder());
        ...
    }
}

或者更徹底一些按傅,直接由OrderService依賴IOrderManage捉超,Consumer只依賴OrderService。
這種抽象是對IsAvailable邏輯的抽象逞敷。

總之狂秦,業(yè)務(wù)邏輯遷移到了service灌侣。之后我們的單元測試問題可以使用Mock<IOrderService>解決推捐。

歡迎拍磚歡迎反饋

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市侧啼,隨后出現(xiàn)的幾起案子牛柒,更是在濱河造成了極大的恐慌,老刑警劉巖痊乾,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件皮壁,死亡現(xiàn)場離奇詭異,居然都是意外死亡哪审,警方通過查閱死者的電腦和手機蛾魄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來湿滓,“玉大人滴须,你說我怎么就攤上這事∵窗拢” “怎么了扔水?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長朝氓。 經(jīng)常有香客問我魔市,道長,這世上最難降的妖魔是什么赵哲? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任待德,我火速辦了婚禮,結(jié)果婚禮上枫夺,老公的妹妹穿的比我還像新娘将宪。我一直安慰自己,他們只是感情好筷屡,可當我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布涧偷。 她就那樣靜靜地躺著,像睡著了一般毙死。 火紅的嫁衣襯著肌膚如雪燎潮。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天扼倘,我揣著相機與錄音确封,去河邊找鬼除呵。 笑死,一個胖子當著我的面吹牛爪喘,可吹牛的內(nèi)容都是我干的颜曾。 我是一名探鬼主播,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼秉剑,長吁一口氣:“原來是場噩夢啊……” “哼泛豪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起侦鹏,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤诡曙,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后略水,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體价卤,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年渊涝,在試婚紗的時候發(fā)現(xiàn)自己被綠了慎璧。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡跨释,死狀恐怖胸私,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情煤傍,我是刑警寧澤盖文,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站蚯姆,受9級特大地震影響五续,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜龄恋,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一疙驾、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧郭毕,春花似錦它碎、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至乘碑,卻和暖如春挖息,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背兽肤。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工套腹, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留绪抛,地道東北人。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓电禀,卻偏偏與公主長得像幢码,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子尖飞,可洞房花燭夜當晚...
    茶點故事閱讀 44,979評論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理症副,服務(wù)發(fā)現(xiàn),斷路器葫松,智...
    卡卡羅2017閱讀 134,659評論 18 139
  • Mock 方法是單元測試中常見的一種技術(shù)瓦糕,它的主要作用是模擬一些在應(yīng)用中不容易構(gòu)造或者比較復雜的對象,從而把測試與...
    熊熊要更努力閱讀 28,356評論 2 25
  • 本文作者:張樂腋么。全文約 4199 字,讀完可能需要 7 分鐘亥揖。雖然這篇不是以 Python 為示例的珊擂,但基本的思路...
    羅義的夏天閱讀 924評論 0 3
  • 坐標重慶,大一黨费变,喜歡看書摧扇,寫點文章,記錄生活中的喜怒哀樂挚歧。有很多喜歡和想做的事情扛稽,繪畫,學吉他滑负,學韓語在张,旅游,而...
    卷發(fā)元元閱讀 1,336評論 0 3
  • 有人說拱橋便是個輪回矮慕。有人匆匆而過帮匾,有人駐足欣賞,走過的也是人生痴鳄。天上一半水里一半瘟斜,畫面變得圓滿;真一半假一半痪寻,不...
    大胡子張閱讀 244評論 0 0