dotnet 代碼優化 聊聊邏輯圈複雜度

来源:https://www.cnblogs.com/lindexi/archive/2023/01/09/17038319.html
-Advertisement-
Play Games

移動滑鼠到你想要的位置,然後進行點擊,某些時候是很有用的 using System; using System.Drawing; using System.Runtime.InteropServices; using System.Windows.Forms; private void button ...


本文屬於 dotnet 代碼優化系列博客。相信大家都對圈複雜度這個概念很是熟悉,本文來和大家聊聊邏輯的圈複雜度。代碼優化裡面,一個關註的重點在於代碼的邏輯複雜度。一段代碼的邏輯複雜度越高,那麼維護起來的難度也就越大。衡量代碼的邏輯複雜度的一個維度是通過邏輯圈複雜度進行衡量。本文將告訴大家如何判斷代碼的邏輯圈複雜度以及一些降低圈複雜度的套路,讓大家瞭解如何寫出更好維護的代碼

回顧一下代碼設計的目標,其中一個很重要的點就是解決 複雜的代碼邏輯 和 人類有限的智商 的矛盾。假設人類的智商非常的高,無論再複雜的代碼邏輯都能理解,且人類寫出的邏輯也不存在漏洞,那其實很多代碼設計都是不需要的。現實剛好不是,一個稍微複雜的項目,就已經不是人類輕而易舉能夠掌控的。即使是自己編寫的代碼,也會隨著時間逐漸遺忘代碼裡面當初的實現邏輯。何況在團隊協作中,可能會遇到需要閱讀其他開發者留下的代碼的時候,假設前輩們沒有好好的進行編寫和設計,自然可能是給後來者挖了一個大坑

邏輯的圈複雜度屬於一個度量代碼複雜度的維度,但稍微特別的是,當邏輯的圈複雜度比較低時,能意味著代碼複雜度比較低,比較好維護。但反過來不成立,比較好維護的代碼,不一定是邏輯的圈複雜度比較低的代碼。代碼的可維護是需要綜合考慮多個維度的,雖然說降低邏輯的圈複雜度基本上都是屬於正確的事情,但由於實際項目遇到的情況比較特殊,還請識別主次矛盾,不要強行優化

邏輯的圈複雜度是指在代碼執行過程中,邏輯上形成的圈的數量,更多的是指在面向對象設計裡面的類和方法之間的關係。至於方法內的迴圈判斷等,只屬於(代碼)圈複雜度(Cyclomatic complexity)而不是邏輯圈複雜度

學術的定義,相信大家都不感興趣,下麵來舉一個例子,相信大家看完很快就懂了

例子依然是老套的圖書管理系統的故事,假定書籍有 人文、哲學、物理、數學、電腦 等類型的書籍,在圖書管理系統裡面,需要有一定的業務邏輯,對其進行處理。其工序有些是所有類型共用的,有些是需要根據類型而來的,假定每個工序都能用一個代碼方法完成。原始的邏輯設計抽象起來如下圖

從邏輯上看,以上的邏輯設計是存在很多個圈圈的,相當於不停的拆分、聚合,每一次都是在增加邏輯圈複雜度,這樣的邏輯設計對應到代碼裡面,大概就是一堆 if 或者 switch 判斷,控制其後續走向,或者是面向對象的繼承關係,讓調用穿插在基類和子類之間。假定以上的邏輯設計屬於使用了 一堆 if 或者 switch 判斷的方式,那自然在區分輸入類型和工序1裡面,都會存在判斷書籍類型,以調用後續邏輯的代碼,偽代碼如下

void 區分輸入類型()
{
    if (書籍類型 == 人文)
    {
        人文_工序0();
    }
    else if (書籍類型 == 哲學)
    {
        哲學_工序0();
    }
    else if(...)
    {
        ...
    }
}

void 人文_工序0()
{
    // 工序的邏輯
    ...

    不分圖書類型的_工序1();
}

void 哲學_工序0()
{
    // 工序的邏輯
    ...

    不分圖書類型的_工序1();
}

void 不分圖書類型的_工序1()
{
    // 工序的邏輯
    ...

    if (書籍類型 == 人文)
    {
        人文_工序2();
    }
    else if (書籍類型 == 哲學)
    {
        哲學_工序2();
    }
    else if(...)
    {
        ...
    }
}

...

從以上的偽代碼也可以看到,在 區分輸入類型不分圖書類型的_工序1 之間,存在邏輯比較相似的代碼,那就是拆分書籍類型,然後調用不同的方法。當書籍的類型足夠多的時候,這個邏輯維護起來就開始令人煩躁起來了,當工序同樣多起來的時候,那就更加不好玩咯

來數數邏輯的圈圈數量,猜猜有多少個圈圈?如下圖標記出來的只有 4 個圈圈對不

其實沒有那麼簡單。嗯,不嚴謹的算,上面的邏輯設計圖至少有 9 個圈圈

如果列出更多的書籍類型,以及更多的工序,那這個圈的數量能夠更加龐大

大家也可以想想看,每加一個書籍類型,會加多少個圈圈?世界上還有一群專家也在研究加一個模塊或一個功能時,圈複雜度的增加速率。在某些時候的設計上,會導致加一個模塊或加一個功能時,增加的圈圈數量會越來越多。例如上面的邏輯設計圖在兩個書籍類型,也就是兩個模塊時,只有三個圈圈,但是在有三個模塊時,就有 9 個圈圈了。也可以看到,隨著書籍類型的數量,也就是模塊的數量,不斷增加的時候,每加一個時,增加的圈圈數量會越來越多,這也就表示了邏輯複雜度每次增加都會越來越多

換一句話說,如果按照上面的邏輯設計圖的方式進行開發,會發現越開發越複雜。即使開發者有著很好的編寫代碼的能力,也會逐漸發現整個項目越來越難以掌控。在設計上存在將會導致必然出現的代碼邏輯圈複雜度時,會導致項目在開發過程中是上帝和程式猿才能看懂代碼,開發一定時間之後,就只有上帝才能看懂代碼了

在瞭解基礎的知識之後,大家也許會問,那如何改造降低圈複雜度呢?一個套路方法就是在區分類型之後,讓數據的走向被具體類型進行控制,這也是面向對象里,多態的一個用法。具體來做就是在 區分輸入類型 的類型之後,進入某個類型的書籍的總處理方法,在某個類型的總處理方法裡面,可以愉快的從工序的開始執行到工序的結束

再來數一下邏輯的圈複雜度,是不是一個圈也數不到了?對應的代碼大概如下,可以看到每個總工序裡面處理的邏輯一目瞭然

void 人文_總工序()
{
    人文_工序0();
    工序1();
    人文_工序2();
    工序3();
}

void 哲學_總工序()
{
    哲學_工序0();
    工序1();
    哲學_工序2();
    工序3();
}

啥都不用說,對比代碼量就知道,看代碼的清晰程度也能看起來降低圈複雜度之後的優化

那這時,也許有伙伴說,如果各個總工序都十分相似,是不是也可以再抽一下?是的,但是也需要看情況,如果少部分的重覆邏輯可以帶來更多的代碼清晰度,那這部分的邏輯留著也是可以接受的。但如果在抽一下基礎類型之後,發現邏輯依然清晰,那就開乾吧,畢竟重覆的邏輯也不是什麼好的事情

定義一個書籍處理的抽象基類,然後在此基類裡面放總工序,接著各個具體的書籍處理類型,繼承基類,編寫實現方法,偽代碼如下

abstract class 書籍管理基類
{
    public void 總工序()
    {
        工序0();
        工序1();
        工序2();
        工序3();
    }

    protected abstract void 工序0();

    private void 工序1()
    {
        // ...
    }
    protected abstract void 工序2();

    private void 工序3()
    {
        // ...
    }
}

class 人文書籍管理 : 書籍管理基類
{
    protected override void 工序0()
    {
        人文_工序0();
    }

    private void 人文_工序0()
    {
        // ...
    }

    protected override void 工序2()
    {
        人文_工序2();
    }

    private void 人文_工序2()
    {
        // ...
    }
}

class 哲學書籍管理 : 書籍管理基類
{
    protected override void 工序0()
    {
        哲學_工序0();
    }

    private void 哲學_工序0()
    {
        // ...
    }

    protected override void 工序2()
    {
        哲學_工序2();
    }

    private void 哲學_工序2()
    {
        // ...
    }
}

可以看到,這大概也就是一個超級簡單的框架了,具備了一定的擴展性,也就是後續如果還需要加上新的書籍類型,也是非常方便的,只需要定義多一個類型即可,同時邏輯上也相對來說比較清真,沒有那麼複雜

以上是藉助 C# 裡面的抽象類實現的,這個套路需要不斷讓子類型進行重寫方法,導致邏輯上可能部分是在基類,部分是在子類。不過以上的代碼寫法是沒有問題的,因為繼承關係才只有兩層,但如果繼承關係更多了呢?假設有三層甚至更高呢?這時執行邏輯可能需要跨越多個類型,那邏輯複雜度也會上來

假定有如下圖的邏輯,需要按照順序或者是執行時間,分別調用方法1到6來完成業務端的任務。當存在讓子類型層層繼承的基類有三個的時候,如果調用方法散落在這個基類裡面,那邏輯複雜度將會是非常高的,很多時候靜態閱讀代碼都非常有難度

如上圖,假設以上沒有畫出來圖,而是寫成代碼,那想要靜態閱讀代碼,瞭解其中的執行邏輯,預計看了一會開始亂了,不知道對應的方法應該在哪個類型裡面,哪個文件裡面。好在 C# 裡面禁用了多類型繼承,否則能寫出連示意圖畫出來都能勸退人的代碼。可是 C# 裡面也有一個叫虛方法的定義,允許在基類裡面定義虛方法,看子類的心情去進行重寫,有重寫就使用子類的,沒重寫就採用基類的,上圖裡面的方法 6 是一個虛方法,在基類 2 裡面定義,但是在 基類 3 被重寫。這時將會發現靜態閱讀的代碼,不見得就是實際運行的代碼。例如閱讀到基類 2 裡面定義了方法 6 的邏輯,然而實際運行的時候,執行的是基類 3 的邏輯

這裡需要補充一點的是靜態閱讀代碼指的是和調試閱讀代碼相對的閱讀代碼方式,指的是在不開始進行調試的方式進行閱讀代碼,可以在 IDE 的輔助下,例如在 VisualStudio 這樣的 IDE 輔助下閱讀代碼。好維護的代碼是需要考慮靜態閱讀代碼的,因為很多時候調試的時候能跑的路徑不會特別全,也不會特別多,甚至有些邏輯是存在很多前置條件的,僅靠調試來瞭解執行方式,可能瞭解到不全面

這也是某些開發老司機會說的“組合優於繼承”的其中一點原因,大量的繼承將會導致邏輯散落在各地,不夠“內聚”導致邏輯複雜度上升。值得一提是 “組合優於繼承” 這句話是具備大量前提的,還請不要將這句話作為開發的規範

那什麼時候應該選擇什麼方法?其實十分主觀,我的推薦是多試試看,寫多了,然後將自己坑多了,自然就知道了。主動去看自己之前寫過的複雜邏輯(最好別去看別人的,否則心態可能會炸)看看是否會感覺自己無法理解邏輯,如果會的話,再想想可以使用什麼方式,如果再寫一次的話,可以更加方便閱讀代碼理清邏輯

回顧一下,本文告訴了大家什麼是代碼邏輯圈複雜度,以及降低邏輯圈複雜度的套路方法。同時也告訴了大家,這個套路也不是萬能的,做的不好也可以提升代碼複雜度

更多代碼編寫相關博客,請參閱我的 博客導航

特別感謝 小方 幫忙改正

博客園博客只做備份,博客發佈就不再更新,如果想看最新博客,請到 https://blog.lindexi.com/

知識共用許可協議
本作品採用知識共用署名-非商業性使用-相同方式共用 4.0 國際許可協議進行許可。歡迎轉載、使用、重新發佈,但務必保留文章署名[林德熙](https://www.cnblogs.com/lindexi)(包含鏈接:https://www.cnblogs.com/lindexi ),不得用於商業目的,基於本文修改後的作品務必以相同的許可發佈。如有任何疑問,請與我[聯繫](mailto:[email protected])。
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 今天給大家介紹的是Python爬蟲批量下載評書音頻並保存本地,在這裡給需要的小伙伴們代碼,並且給出一點小心得。 首先是爬取之前應該儘可能偽裝成瀏覽器而不被識別出來是爬蟲,基本的是加請求頭,但是這樣的純文本數據爬取的人會很多,所以我們需要考慮更換代理IP和隨機更換請求頭的方式來對評書精選音頻進行 ...
  • 互聯網世界里最流行的開源關係型資料庫之一就是MySQL/MariaDB了,由於高度的相似,故而直接使用mysql統一指稱。 ...
  • 項目地址:https://github.com/pikeduo/TXTReader PyQt5中文手冊:https://maicss.gitbook.io/pyqt-chinese-tutoral/pyqt5/ QtDesigner學習地址:https://youcans.blog.csdn.net ...
  • 14. 最長公共首碼 題目描述 編寫一個函數來查找字元串數組中的最長公共首碼。 如果不存在公共首碼,返回空字元串 ""。 方法 暴力演算法 先判斷字元串數組是否有為空,為空直接返回空 令第一個字元串作為基準進行比較 設置一個長度,作為最後最長公共首碼的長度 迴圈判斷,選取最小長度 代碼 package ...
  • 2023-01-09 一、在IDEA中創建Maven版的web工程 (1)步驟: ①創建一個maven模塊,命名為“maven_web_end”,之後需要創建web工程的目錄。在“maven_web_end.src.main”下創建“webapp”文件夾(命名必須為webapp,否則識別不了);在“ ...
  • 作者:小牛呼嚕嚕 | https://xiaoniuhululu.com 電腦內功、JAVA底層、面試相關資料等更多精彩文章在公眾號「小牛呼嚕嚕 」 大家好,我是呼嚕嚕,這次我們一起來看看Java記憶體區域,本文 基於HotSpot 虛擬機,JDK8, 乾貨滿滿 前言 Java 記憶體區域, 也叫運行 ...
  • Excelize 是 Go 語言編寫的用於操作 Office Excel 文檔基礎庫,支持 XLAM / XLSM / XLSX / XLTM / XLTX 等多種文檔格式。2023年1月9日,社區正式發佈了 2.7.0 版本,該版本包含了多項新增功能、錯誤修複和相容性提升優化。 ...
  • ThreadLocal是一個關於創建線程局部變數的類。 通常情況下,我們創建的變數是可以被任何一個線程訪問並修改的。而使用ThreadLocal創建的變數只能被當前線程訪問,其他線程則無法訪問和修改。ThreadLocal在設計之初就是為解決併發問題而提供一種方案,每個線程維護一份自己的數據,達到線... ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...