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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...