巧用遞歸解決矩陣最大序列和問題

来源:https://www.cnblogs.com/xiekun/archive/2019/11/05/11802596.html
-Advertisement-
Play Games

之前同事問了一道需要點腦洞的演算法題,我覺得蠻有意思的,思路可能會給大家帶來一些啟發,特意在此記錄一下 題目 現有一個元素僅為 0,1 的 n 階矩陣,求連續相鄰(水平或垂直,不能有環)元素值為 1 的序列和的最大值 假設有如下矩陣 則此矩陣連續相鄰元素為 1 的序列和分別為 4, 3,(如圖示),可 ...


之前同事問了一道需要點腦洞的演算法題,我覺得蠻有意思的,思路可能會給大家帶來一些啟發,特意在此記錄一下

題目

現有一個元素僅為 0,1 的 n 階矩陣,求連續相鄰(水平或垂直,不能有環)元素值為 1 的序列和的最大值
假設有如下矩陣

則此矩陣連續相鄰元素為 1 的序列和分別為 4, 3,(如圖示),可知這個矩陣序列和的最大值為 4

解題思路

要算序列和的最大值,我們可以先找出所有可能的序列和,然後取其中的最大值,那怎麼找這些序列呢?
首先我們發現,每個序列的起點和終點必然是 1,我們可以遍歷矩陣的每一個元素,如果元素值為 1,則將其作為序列的起點開始查找所有以這個元素為起點的序列,我們知道序列是可以向垂直和水平方向延伸的,所以我們可以以這個元素為起點,查找它的上下左右值為 1 的元素,再以找到的這些元素為起點,繼續在元素的上下左右查找值為1的元素(遞歸),如果找不到符合條件的值,則序列終止,在遍歷過程中保存每條序列遍歷的元素,即可知曉每條序列的元素和,從而求得序列和的最大值

文字說得有點繞,接下來我們就以查找以下矩陣的最大序列和為例來詳細看一下如何查找最大序列和

  1. 從左到右,從上到下遍歷所有值為 1 的元素,第一個符合條件的元素在右上角,所以以這個元素為起點來查找序列

  2. 以這個元素為起點,查找這個元素上下左右為值為 1 的元素,發現只有這個元素下麵的元素符合條件

  3. 再以這個元素為起點查找這個元素前後左右值為 1 的元素,可以看到這個元素的上
    ,左元素值為 1,左邊的元素顯然符合條件,而上面的元素由於是當前正在遍歷序列中遍歷過的元素,所以不符合條件(假設上面的元素符合條件,會發生什麼?接下來會尋找以上面元素為起始點的序列,又回到了第一步,陷入無限迴圈,所以元素的下一個值為 1 的元素不能是當前正在遍歷的序列中的元素!,這一點是解題的關鍵,務必要註意!)
    由此可知此時符合條件的元素如下紅圈所示

  4. 再尋找此元素上下左右都為 1 的元素,可以看到這個元素的左右下的元素都為 1,根據上一步的分析可知,右元素是當前正在遍歷序列中已遍歷過的元素,所以不符合條件,那麼只剩下左,下元素符合條件

  5. 再次尋找這兩個元素上下左右皆為 1 的元素,可知符合條件的元素為步驟 3 中的紅框元素,由於此元素是當前正在遍歷序列中已遍歷過的元素,所以不符合條件,序列的遍歷到此終止,至此我們可以知道,從右上角元素為起點的序列和的最大值為 4 ,連接遍歷過的元素,如圖示


  6. 同理接下來再按照以上的步驟依次遍歷剩餘的值為 1 的元素,可知以這些元素為起點的序列和的最大值分別為 4, 3, 3, 4(如下圖)




    (紅圈的元素代表序列遍歷的起始點)
  7. 綜上可知,此矩陣連續相鄰值為 1 的元素的序列和的最大值為 4

代碼實現

好了知道瞭解題思路,現在我們來看下代碼該如何實現
首先我們要用一個數據結構來表示矩陣,顯然矩陣用數組表示很合適,這裡我們用一維數組來表示矩陣,Java 代碼如下

public class Matrix {
    /**
     * @param matrix  矩陣
     * @param dimension 代表 dimension 階矩陣
     * @return 矩陣序列的最大值
     */
    private static Integer getMaxSequetialSum(int[] matrix, int dimension) {
        int count = matrix.length;      // 矩陣的元素個數
        int maxSequentialSum = 0;       // 矩陣序列的最大值
        // 逐個遍歷元素
        for (int index = 0; index < count; index++) {
            int elementValue = matrix[index];
            // 如果當前元素為1,則以此元素為起點,查找以此元素為起點的序列的和的最大值
            if (elementValue == 1) {
                // 記錄以下標為 index 的元素為起點的序列遍歷過的元素位置,以防元素被重覆遍歷
                Set<Integer> traverseElementSet = new HashSet<>();
                traverseElementSet.add(index);
                // 以下標值為 index 的元素為起點的序列的最大值
                int currentSequetialSum = getCurrentVerticeSequetialSum(matrix, traverseElementSet, index, dimension);
                maxSequentialSum = Math.max(maxSequentialSum, currentSequetialSum);
            }
        }
        return maxSequentialSum;
    }

    /**
     * @param matrix  矩陣
     * @param traverseElementSet 序列中已遍歷過的元素的位置
     * @param index     元素的位置,序列的起點
     * @param dimension dimension 階矩陣
     * @return 以位置為 index 的元素為起點的序列的最大值
     */
    private static Integer getCurrentVerticeSequetialSum(int[] matrix, Set<Integer> traverseElementSet, int index, int dimension) {
        // 查找 矩陣中位置為 index 的元素上下左右元素對應的位置
        int left = index - 1;
        int right = index + 1;
        int up = index - dimension;
        int down = index + dimension;

        // 以左元素為起點的序列的值
        int leftIndexSum = 0;

        // 以右元素為起點的序列的值
        int rightIndexSum = 0;

        // 以上元素為起點的序列的值
        int upIndexSum = 0;

        // 以下元素為起點的序列的值
        int downIndexSum = 0;

        /**
         * 以下四個if else 旨在檢查每一個元素位置的有效性,值必須為 1
         * 需要註意的是元素不能是序列已遍歷過的元素!
         * 如果上下左右元素不合法,則序列終止,打點此遍歷序列的元素和
         */

        if (left >= 0 && matrix[left] == 1 && !traverseElementSet.contains(left)) {
            Set<Integer> leftTraverseElementSet = new HashSet<>(traverseElementSet);
            leftTraverseElementSet.add(left);
            leftIndexSum = getCurrentVerticeSequetialSum(matrix, leftTraverseElementSet, left, dimension);
        } else {
            leftIndexSum = traverseElementSet.size();
        }

        // 右元素必須與位置為index的元素在同一行上
        if (right / dimension == index / dimension && matrix[right] == 1 && !traverseElementSet.contains(right)) {
            traverseElementSet.add(right);
            Set<Integer> rightTraverseElementSet = new HashSet<>(traverseElementSet);
            rightTraverseElementSet.add(right);
            rightIndexSum = getCurrentVerticeSequetialSum(matrix, rightTraverseElementSet, right, dimension);
        } else {
            rightIndexSum = traverseElementSet.size();
        }

        if (up >= 0 && matrix[up] == 1 && !traverseElementSet.contains(up)) {
            Set<Integer> upTraverseElementSet = new HashSet<>(traverseElementSet);
            upTraverseElementSet.add(up);
            upIndexSum = getCurrentVerticeSequetialSum(matrix, upTraverseElementSet, up, dimension);
        } else {
            upIndexSum = traverseElementSet.size();
        }

        if (down < matrix.length && matrix[down] == 1 && !traverseElementSet.contains(down)) {
            Set<Integer> downTraverseElementSet = new HashSet<>(traverseElementSet);
            downTraverseElementSet.add(down);
            downIndexSum = getCurrentVerticeSequetialSum(matrix, downTraverseElementSet, down, dimension);
        } else {
            downIndexSum = traverseElementSet.size();
        }

        // 查找以位置為 index 的元素為起點各向上下左右延伸的序列的最大值
        return Collections.max(Arrays.asList(leftIndexSum, rightIndexSum, upIndexSum, downIndexSum));
    }

    public static void main(String[] args) {
        // 初始化矩陣,假設此矩陣為 5 x 5 矩陣
        int[] matrix1 = {
                0,0,0,0,1,
                0,0,1,1,1,
                0,0,0,1,0,
                0,0,0,0,0,
        };
        int max = Matrix.getMaxSequetialSum(matrix1, 5);
        System.out.println(max);  // 列印4

        int[] matrix2 = {
                0,0,0,0,1,
                0,0,1,1,1,
                0,0,1,1,0,
                0,0,0,0,0,
        };
        max = Matrix.getMaxSequetialSum(matrix2, 5);
        System.out.println(max);  // 列印6
    }
}

時間複雜度與空間複雜度分析

任何演算法,如果不談時間複雜度與空間複雜度都是耍流氓,接下來我們看下以上解法的時間複雜度和空間複雜度
1.首先來看空間複雜,由於在在遍歷過程中我們用了記錄遍歷序列元素位置的 traverseElementSet,所以空間複雜度顯然是 O(n)
2.這道題用了遞歸,時間複雜度確實挺複雜的,也比較考驗程式員的水平,直觀上看不出來,那我們看下怎麼推導,我們用 f(n) 來表示以位置為 n 的元素為起點的序列和的計算次數,從以上的推導可知,只要計算出以此元素的上下左右元素為起點的序列和的最大值,也自然知道了 f(n)。即計算以位置 n 為起點的序列和次數換算成計算以此元素的上下左右元素為起點的序列和的次數

f(n) = f(左) + f(右) + f(上) + f(下)

仔細考慮一下可知以上下左右四個元素為起點的序列和的計算次數可以認為是一樣的
從而有
f(n) = 4f(左)
假設矩陣元素個數為N,則
f(n) = 4N
由於有 N 個元素,所以可知總的時間複雜度為 O(4N2),即 O(n2)

總結

這道題乍一看確實沒什麼頭緒,無法像反轉二叉樹那樣比較容易地看出使用遞歸的思路去解決,所以我們需要耐心地去分析,學會把問題分解,分解思路如下
求序列的最大和轉化為求所有序列的和 ----> 轉化成如何找尋所有的序列 ----> 觀察到序列的起點的元素必須是 1 ----> 想到如何找尋以此元素為起點的所有序列 ----> 只要找到以這個元素上下左右值為 1 的元素為起點的所有序列和 ----> 再以上下左右元素值為 1 的元素為起點遞歸找尋以它們各自的上下左右值為 1 的元素為起點的所有序列和 ----> 找到所有的序列和後自然就找到了最大序列和

個人微信號「geekoftaste」,歡迎加微信一起交流,共同進步


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. Self Encapsulate Field(自封裝值域) 你直接訪問一個值域,但與值域之間的耦合關係逐漸變得笨拙。為這個值域建立取值/設值函數(get/set),並且只以這些函數來訪問值域。 應用場景:如果你想訪問superclass中的一個值域,卻又想在subclass中將[對這個變數的訪 ...
  • 1. 創建一個springboot工程 2. 創建一個實體類 @Data //想相當於@Setter、@Getter和@ToString替代了setter、getter、toString方法 @AllArgsConstructor//全參數構造方法 @NoArgsConstructor//無參構造方 ...
  • 01、前言 先讓我吐一句肺腑之言吧,不說出來會憋出內傷的。《Java 併發編程實戰》這本書太特麽枯燥了,儘管它被奉為併發編程當中的經典之作,但我還是忍不住。因為第四章“對象的組合”我整整啃了兩周的時間,才啃出來點肉絲。 讀者朋友們見諒啊。要怪只能怪我自己的學習能力有限,真讀不了這種生硬無趣的技術書。 ...
  • 本文是在學習軟體工程與J2EE課程時的學習筆記,旨在從大體的概念上瞭解Java EE的一些主要組件在Web應用中的作用。 上圖精煉的描述了MVC模型以及Java EE的部分組件如何分佈在一個Web應用上,下文所提到的圖示均指該圖。 Web應用 在開始一切之前要瞭解什麼是Web應用,對於圖中最左側的用 ...
  • FFTW是一個可以進行可變長度一維或多維DFT的開源C程式庫,是目前最快的FFT演算法實現。 本文簡述了在Windows平臺上,如何在C++中調用FFTW,所使用的IDE為Visual Studio 2017。 FFTW的詳細信息可在 http://www.fftw.org 中查看 獲取FFTW 在 ...
  • 一、正則的寫法: . (點好) :表示任意一個字元,除了\n,比如查找所有的一個字元\. [] :匹配中括弧中列舉的任意字元,比如[L,Y,0], LLY, Y0, LIU \d :任意一個數字 \D :除了數字都可以 \s :表示空格,tab鍵 \S :除了空白符號 \w :單詞字元,就是a-z, ...
  • 升級 https 記錄 1、去阿裡雲購買證書(免費版),並提交審核資料 購買的證書 2、下載證書 下載證書 3、查看上圖頁面的第三步 JKS證書安裝 4、在證書目錄下執行阿裡雲提供的命令,密碼都填 pfx password.txt 中的內容(三次),會生成 your name.jks 文件。 生成 ...
  • ▶ Log4j2 性能 "https://logging.apache.org/log4j/2.x/performance.html" ▶ Spring Boot 依賴與配置 Maven 依賴 XML 配置 resources/log4j2.xml 混合 sync/async 彩色日誌 分類輸出到不 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...