Android 啟動優化(二) - 有向無環圖的原理以及解題思路

来源:https://www.cnblogs.com/gdutxiaoxu/archive/2023/02/22/17145999.html
-Advertisement-
Play Games

Android 啟動優化(一) - 有向無環圖 Android 啟動優化(二) - 拓撲排序的原理以及解題思路 Android 啟動優化(三) - AnchorTask 使用說明 Android 啟動優化(四)- 手把手教你實現 AnchorTask Android 啟動優化(五)- AnchorT ...


Android 啟動優化(一) - 有向無環圖

Android 啟動優化(二) - 拓撲排序的原理以及解題思路

Android 啟動優化(三) - AnchorTask 使用說明

Android 啟動優化(四)- 手把手教你實現 AnchorTask

Android 啟動優化(五)- AnchorTask 1.0.0 版本更新了

Android 啟動優化(六)- 深入理解佈局優化

這幾篇文章從 0 到 1,講解 DAG 有向無環圖是怎麼實現的,以及在 Android 啟動優化的應用。

推薦理由:現在挺多文章一談到啟動優化,動不動就聊拓撲結構,這篇文章從數據結構到演算法、到設計都給大家說清楚了,開源項目也有非常強的借鑒意義。

前言

春節之前,更新了一篇博客 Android 啟動優化(一) - 有向無環圖,反響還不錯,今天,讓我們一起來看一下,怎樣用代碼實現有向無環圖。

基本概念

拓撲排序的英文名是 Topological sorting。

拓撲排序要解決的問題是給一個圖的所有節點排序。有向無環圖才有拓撲排序,非有向無環圖沒有。

換句話說,拓撲排序必須滿足以下條件

圖必須是一個無環有向圖。序列必須滿足的條件:

  • 每個頂點出現且只出現一次。
  • 若存在一條從頂點 A 到頂點 B 的路徑,那麼在序列中頂點 A 出現在頂點 B 的前面。

實戰

我們已 leetcode 上面的一道演算法題目作為切入點進行講解。

leeocode 210: https://leetcode-cn.com/problems/course-schedule-ii/

eg: 現在你總共有 n 門課需要選,記為 0 到 n-1。

在選修某些課程之前需要一些先修課程。 例如,想要學習課程 0 ,你需要先完成課程 1 ,我們用一個匹配來表示他們: [0,1]

給定課程總量以及它們的先決條件,返回你為了學完所有課程所安排的學習順序。

可能會有多個正確的順序,你只要返回一種就可以了。如果不可能完成所有課程,返回一個空數組。

示例 1

輸入: 2, [[1,0]] 
輸出: [0,1]
解釋: 總共有 2 門課程。要學習課程 1,你需要先完成課程 0。因此,正確的課程順序為 [0,1] 。

示例 2

輸入: 4, [[1,0],[2,0],[3,1],[3,2]]
輸出: [0,1,2,3] or [0,2,1,3]
解釋: 總共有 4 門課程。要學習課程 3,你應該先完成課程 1 和課程 2。並且課程 1 和課程 2 都應該排在課程 0 之後。
     因此,一個正確的課程順序是 [0,1,2,3] 。另一個正確的排序是 [0,2,1,3] 。

這道題,很明顯,看起來可以有有向無環圖的解法來解決

BFS 演算法

題目分析

我們首先引入有向圖 描述依賴關係

示例:假設 n = 6,先決條件表:[ [3, 0], [3, 1], [4, 1], [4, 2], [5, 3], [5, 4] ]

  • 0, 1, 2 沒有先修課,可以直接選。其餘的,都要先修 2 門課
  • 我們用 有向圖 描述這種 依賴關係 (做事的先後關係):

在有向圖中,我們知道,有入度出度概念:

如果存在一條有向邊 A --> B,則這條邊給 A 增加了 1 個出度,給 B 增加了 1 個入度。所以頂點 0、1、2 的 入度為 0。 頂點 3、4、5 的 入度為 2

BFS 前準備工作

  • 我們關心 課程的入度 —— 該值要被減,要被監控
  • 我們關心 課程之間的依賴關係 —— 選這門課會減小哪些課的入度
  • 因此我們需要合適的數據結構,去存儲這些關係,這個可以通過哈希表

解題思路

  • 維護一個 queue,裡面都是入度為 0 的課程
  • 選擇一門課,就讓它出列,同時 查看哈希表,看它 對應哪些後續課
  • 將這些後續課的 入度 - 1,如果有減至 0 的,就將它推入 queue
  • 不再有新的入度 0 的課入列 時,此時 queue 為空,退出迴圈
    private  class Solution {
        public int[] findOrder(int num, int[][] prerequisites) {

            // 計算所有節點的入度,這裡用數組代表哈希表,key 是 index, value 是 inDegree[index].實際開發當中,用 HashMap 比較靈活
            int[] inDegree = new int[num];
            for (int[] array : prerequisites) {
                inDegree[array[0]]++;
            }

            // 找出所有入度為 0 的點,加入到隊列當中
            Queue<Integer> queue = new ArrayDeque<>();
            for (int i = 0; i < inDegree.length; i++) {
                if (inDegree[i] == 0) {
                    queue.add(i);
                }
            }
            
            ArrayList<Integer> result = new ArrayList<>();
            while (!queue.isEmpty()) {
                Integer key = queue.poll();
                result.add(key);
                // 遍歷所有課程
                for (int[] p : prerequisites) {
                    // 改課程依賴於當前課程 key
                    if (key == p[1]) {
                        // 入度減一
                        inDegree[p[0]]--;
                        if (inDegree[p[0]] == 0) {
                            queue.offer(p[0]); // 加入到隊列當中
                        }
                    }
                }
            }

            // 數量不相等,說明存在環
            if (result.size() != num) {
                return new int[0];
            }

            int[] array = new int[num];
            int index = 0;
            for (int i : result) {
                array[index++] = i;

            }

            return array;
        }
    }

DFS 解法

演算法思想

  • 對圖執行深度優先搜索。
  • 在執行深度優先搜索時,若某個頂點不能繼續前進,即頂點的出度為0,則將此頂點入棧。
  • 最後得到棧中順序的逆序即為拓撲排序順序。
// 方法 2:鄰接矩陣 + DFS   由於用的數組,每次都要遍歷,效率比較低
    public int[] findOrder(int numCourses, int[][] prerequisites) {
        if (numCourses == 0) return new int[0];
        // 建立鄰接矩陣
        int[][] graph = new int[numCourses][numCourses];
        for (int[] p : prerequisites) {
            graph[p[1]][p[0]] = 1;
        }
        // 記錄訪問狀態的數組,訪問過了標記 -1,正在訪問標記 1,還未訪問標記 0
        int[] status = new int[numCourses];
        Stack<Integer> stack = new Stack<>();  // 用棧保存訪問序列
        for (int i = 0; i < numCourses; i++) {
            if (!dfs(graph, status, i, stack)) return new int[0]; // 只要存在環就返回
        }
        int[] res = new int[numCourses];
        for (int i = 0; i < numCourses; i++) {
            res[i] = stack.pop();
        }
        return res;
    }

    private boolean dfs(int[][] graph, int[] status, int i, Stack<Integer> stack) {
        if (status[i] == 1) return false; // 當前節點在此次 dfs 中正在訪問,說明存在環
        if (status[i] == -1) return true;

        status[i] = 1;
        for (int j = 0; j < graph.length; j++) {
            // dfs 訪問當前課程的後續課程,看是否存在環
            if (graph[i][j] == 1 && !dfs(graph, status, j, stack)) return false;
        }
        status[i] = -1;  // 標記為已訪問
        stack.push(i);
        return true;
    }

總結

這篇博客從實戰的角度出發,介紹了有向無環圖的兩種解法,入度表法和 DFS 法。其中,入度表法很重要,必須掌握。下一篇,我們將從 項目實戰的角度來講解,怎樣搭建一個有向無環圖的通用框架,敬請期待。

ps

AnchorTask 源碼已經更新到 github,AnchorTask,下一篇,將輸出 AnchorTask 使用說明,敬請期待。

如果你覺得對你有所幫助,可以關註我的微信公眾號徐公


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

-Advertisement-
Play Games
更多相關文章
  • window系統下 按此鍵 執行此操作 Command + Shift + B 顯示或隱藏收藏夾欄 Command + Shift + C 打開開發人員工具 Command + D 將當前選項卡另存為收藏夾 Command + Shift + D 在新文件夾中將所有打開的標簽頁另存為收藏夾 Comm ...
  • 需求:批量獲取文本指定內容所在行以下內容(含當前行) 解決方案:使用Powershell腳本處理 案例: 獲取當前文件夾下所有txt文件 含文本"4"所在行以下內容(含當前行) 如果有多行包含文本"4",取第一個所在行以下內容(含當前行) 1.查看當前文件夾內容 2.右鍵執行腳本刪除文件指定內容所在 ...
  • 環境 CPU:Phytium,S2500/64 C00 內核版本:4.19.90-25.10 網訊網卡:txgbe 共兩台設備,光纖直連 復現步驟 設備A、B分別執行以下操作,即可復現 modprobe fcoe systemctl start lldpad systemctl start fcoe ...
  • 本部分介紹可編程並行介面晶元8255A&&可編程定時器、計時器晶元8253、8254,增加了一些具體系統的設計案例。 ...
  • 摘要:ChatGPT承認了自己背後使用的資料庫是Cassandra。 OpenAI最近發佈的AI驅動的智能聊天機器人ChatGPT在互聯網上掀起了一陣風暴,熱衷於嘗試這一新AI成果的網民不在少數。ChatGPT針對網友廣泛的問題提供了非常有針對性的回答,其不可思議的能力成為各大媒體平臺的頭條新聞,其 ...
  • SQL的分類 DDL:數據定義語言 CREATE\ALTER\RENAME(重命名)\DROP\TRUNCATE(清空表) DML:數據操作語言 INSERT\DELETE\UPDATE\SELECT(增刪改查) DCL:數據控制語言 COMMIT(提交)\ROLLBACK(回滾)\SAVEPOIN ...
  • 如今,各行各業都已經意識到了數據的價值,開始沉澱數據資產,挖掘數據價值,但是數據本身其實是很難直觀地看到其價值的。數據就是存儲在電腦系統的“01”代碼,如果你不去用它,能有什麼價值? 正如美國哈佛大學教授格林先生所說:數據本身並不等於知識,更不是智慧,只有經過正確分析之後,數據才能凸顯它的意義。 ...
  • 小編提醒:拿MariaDB的so去MySQL里install,這種方式很容易導致 audit plugin工作異常,不推薦這麼做。強烈建議使用GreatSQL,自帶 audit plugin。 前言 資料庫審計功能主要將用戶對資料庫的各類操作行為記錄審計日誌,以便日後進行跟蹤、查詢、分析,以實現對用 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...