【算法優選】 動態規劃之路徑問題——貳

文章目錄

  • 🎋前言
  • 🌲[下降最小路徑和](https://leetcode.cn/problems/minimum-path-sum/)
    • 🚩題目描述
    • 🚩算法思路:
    • 🚩代碼實現
  • 🎍[最小路徑和](https://leetcode.cn/problems/minimum-path-sum/)
    • 🚩算法思路
    • 🚩代碼實現
  • 🌴[地下城游戲](https://leetcode.cn/problems/dungeon-game/)
    • 🚩題目描述
    • 🚩算法思路
    • 🚩代碼實現
  • ?總結

🎋前言

動態規劃相關題目都可以參考以下五個步驟進行解答:

  1. 狀態表?

  2. 狀態轉移?程

  3. 初始化

  4. 填表順序

  5. 返回值

后面題的解答思路也將按照這五個步驟進行講解。

🌲下降最小路徑和

🚩題目描述

給你一個 n x n 的 方形 整數數組 matrix ,請你找出并返回通過 matrix 的下降路徑 的 最小和 。

下降路徑 可以從第一行中的任何元素開始,并從每一行中選擇一個元素。在下一行選擇的元素和當前行所選元素最多相隔一列(即位于正下方或者沿對角線向左或者向右的第一個元素)。具體來說,位置 (row, col) 的下一個元素應當是 (row + 1, col - 1)、(row + 1, col) 或者 (row + 1, col + 1) 。

  • 示例 1:

在這里插入圖片描述
輸入:matrix = [[2,1,3],[6,5,4],[7,8,9]]
輸出:13
解釋:如圖所示,為和最小的兩條下降路徑

  • 示例 2:

在這里插入圖片描述
輸入:matrix = [[-19,57],[-40,-5]]
輸出:-59
解釋:如圖所示,為和最小的下降路徑

class Solution {public int minFallingPathSum(int[][] matrix) {}
}

🚩算法思路:

關于這?類題,由于我們做過類似的,因此「狀態表?」以及「狀態轉移」是?較容易分析出來的。
?較難的地?可能就是對于「邊界條件」的處理。

  1. 狀態表?:
    對于這種「路徑類」的問題,我們的狀態表??般有兩種形式:
    • 從 [i, j] 位置出發,到達?標位置有多少種?式;
    • 從起始位置出發,到達 [i, j] 位置,?共有多少種?式

這?選擇第?種定義狀態表?的?式:
dp[i][j] 表?:到達 [i, j] 位置時,所有下降路徑中的最?和。

  1. 狀態轉移?程:
    對于普遍位置 [i, j] ,根據題意得,到達 [i, j] 位置可能有三種情況:
    • 從正上? [i - 1, j] 位置轉移到 [i, j] 位置;
    • 從左上? [i - 1, j - 1] 位置轉移到 [i, j] 位置;
    • 從右上? [i - 1, j + 1] 位置轉移到 [i, j] 位置;

我們要的是三種情況下的「最?值」,然后再加上矩陣在 [i, j] 位置的值。于是 dp[i][j] = min(dp[i - 1][j], min(dp[i - 1][j - 1], dp[i - 1][j +1])) + matrix[i][j] 。

  1. 初始化:
    可以在最前?加上?個「輔助結點」,幫助我們初始化。使?這種技巧要注意兩個點:
    • 輔助結點??的值要「保證后續填表是正確的」;
    • 「下標的映射關系」。

在本題中,需要「加上??」,并且「加上兩列」。所有的位置都初始化為?窮?,然后將第??初始化為0 即可。

  1. 填表順序:
    根據「狀態表?」,填表的順序是「從上往下」。

  2. 返回值:
    注意這?不是返回 dp[m][n] 的值!

題?要求「只要到達最后??」就?了,因此這?應該返回「dp表中最后??的最?值」。

🚩代碼實現

class Solution {public int minFallingPathSum(int[][] matrix) {// 1. 創建 dp 表// 2. 初始化// 3. 填表// 4. 返回結果int n = matrix.length;int[][] dp = new int[n + 1][n + 2];for(int i = 1; i <= n; i++) {dp[i][0] = dp[i][n + 1] = Integer.MAX_VALUE;}for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++) {dp[i][j] = Math.min(dp[i - 1][j], Math.min(dp[i - 1][j - 1],dp[i - 1][j + 1])) + matrix[i - 1][j - 1];}}int ret = Integer.MAX_VALUE;for(int j = 1; j <= n; j++) {ret = Math.min(ret, dp[n][j]);}return ret;}
}

🎍最小路徑和

給定一個包含非負整數的 m x n 網格 grid ,請找出一條從左上角到右下角的路徑,使得路徑上的數字總和為最小。

說明:每次只能向下或者向右移動一步。

  • 示例 1:
    在這里插入圖片描述
    輸入:grid = [[1,3,1],[1,5,1],[4,2,1]]
    輸出:7
    解釋:因為路徑 1→3→1→1→1 的總和最小。

  • 示例 2:
    輸入:grid = [[1,2,3],[4,5,6]]
    輸出:12

class Solution {public int minPathSum(int[][] grid) {}
}

🚩算法思路

像這種表格形式的動態規劃,是?常容易得到「狀態表?」以及「狀態轉移?程」的,可以歸結到「不同路徑」?類的題??。

  1. 狀態表?:
    對于這種路徑類的問題,我們的狀態表??般有兩種形式:
    • 從 [i, j] 位置出發,一系列操作;
    • 從起始位置出發,到達 [i, j] 位置,一系列操作。

這?選擇第?種定義狀態表?的?式:dp[i][j] 表?:到達 [i, j] 位置處,最?路徑和是多少。

  1. 狀態轉移:
    簡單分析?下。如果 dp[i][j] 表?到達到達 [i, j] 位置處的最?路徑和,那么到達[i, j] 位置之前的??步,有兩種情況:
    • 從 [i - 1, j] 向下??步,轉移到 [i, j] 位置;
    • 從 [i, j - 1] 向右??步,轉移到 [i, j] 位置。

由于到 [i, j] 位置兩種情況,并且我們要找的是最?路徑,因此只需要這兩種情況下的最?值,再加上 [i, j] 位置上本?的值即可。也就是: dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j]

  1. 初始化:
    可以在最前?加上?個「輔助結點」,幫助我們初始化。使?這種技巧要注意兩個點:
    • 輔助結點??的值要「保證后續填表是正確的」;
    • 「下標的映射關系」。

在本題中,「添加??」,并且「添加?列」后,所有位置的值可以初始化為?窮?,然后讓dp[0][1] = dp[1][0] = 1 即可。

  1. 填表順序:
    根據「狀態轉移?程」的推導來看,填表的順序就是「從上往下」填每??,每??「從左往后」。

  2. 返回值:
    根據「狀態表?」,我們要返回的結果是 dp[m][n]

🚩代碼實現

class Solution {public int minPathSum(int[][] grid) {// 1. 創建 dp 表// 2. 初始化// 3. 填表// 4. 返回值int m = grid.length;int n = grid[0].length;int[][] dp = new int[m + 1][n + 1];for(int j = 0; j <= n; j++)  {dp[0][j] = Integer.MAX_VALUE;}for(int i = 0; i <= m; i++) {dp[i][0] = Integer.MAX_VALUE;}dp[0][1] = dp[1][0] = 0;for(int i = 1; i <= m; i++){for(int j = 1; j <= n; j++) {dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - 1]) + grid[i - 1][j-1];}}return dp[m][n];}
}

🌴地下城游戲

🚩題目描述

惡魔們抓住了公主并將她關在了地下城 dungeon 的 右下角 。地下城是由 m x n 個房間組成的二維網格。我們英勇的騎士最初被安置在 左上角 的房間里,他必須穿過地下城并通過對抗惡魔來拯救公主。

騎士的初始健康點數為一個正整數。如果他的健康點數在某一時刻降至 0 或以下,他會立即死亡。

有些房間由惡魔守衛,因此騎士在進入這些房間時會失去健康點數(若房間里的值為負整數,則表示騎士將損失健康點數);其他房間要么是空的(房間里的值為 0),要么包含增加騎士健康點數的魔法球(若房間里的值為正整數,則表示騎士將增加健康點數)。

為了盡快解救公主,騎士決定每次只 向右 或 向下 移動一步。

返回確保騎士能夠拯救到公主所需的最低初始健康點數。

注意:任何房間都可能對騎士的健康點數造成威脅,也可能增加騎士的健康點數,包括騎士進入的左上角房間以及公主被監禁的右下角房間。

  • 示例 1:
    在這里插入圖片描述
    輸入:dungeon = [[-2,-3,3],[-5,-10,1],[10,30,-5]]
    輸出:7
    解釋:如果騎士遵循最佳路徑:右 -> 右 -> 下 -> 下 ,則騎士的初始健康點數至少為 7 。

  • 示例 2:
    輸入:dungeon = [[0]]
    輸出:1

class Solution {public int calculateMinimumHP(int[][] dungeon) {}
}

🚩算法思路

  1. 狀態表?:

這道題如果我們定義成:從起點開始,到達 [i, j] 位置的時候,所需的最低初始健康點數。那么我們分析狀態轉移的時候會有?個問題:那就是我們當前的健康點數還會受到后?的路徑的影響。也就是從上往下的狀態轉移不能很好地解決問題。

這個時候我們要換?種狀態表?:從 [i, j] 位置出發,到達終點時所需要的最低初始健康點數。這樣我們在分析狀態轉移的時候,后續的最佳狀態就已經知曉。

綜上所述,定義狀態表?為:
dp[i][j] 表?:從 [i, j] 位置出發,到達終點時所需的最低初始健康點數。

  1. 狀態轉移?程:
    對于 dp[i][j] ,從 [i, j] 位置出發,下?步會有兩種選擇(為了?便理解,設 dp[i] [j] 的最終答案是 x ):
    • ?到右邊,然后?向終點
      那么我們在 [i, j] 位置的最低健康點數加上這?個位置的消耗,應該要?于等于右邊位置的最低健康點數,也就是: x + dungeon[i][j] >= dp[i][j + 1] 。通過移項可得: x >= dp[i][j + 1] - dungeon[i][j] 。因為我們要的是最?值,因此這種情況下的 x = dp[i][j + 1] - dungeon[i][j] ;
    • ?到下邊,然后?向終點
      那么我們在 [i, j] 位置的最低健康點數加上這?個位置的消耗,應該要?于等于下邊位置的最低健康點數,也就是: x + dungeon[i][j] >= dp[i + 1][j] 。通過移項可得: x >= dp[i + 1][j] - dungeon[i][j] 。因為我們要的是最?值,因此這種情況下的 x = dp[i + 1][j] - dungeon[i][j] ;

綜上所述,我們需要的是兩種情況下的最?值,因此可得狀態轉移?程為:dp[i][j] = min(dp[i + 1][j], dp[i][j + 1]) - dungeon[i][j]

但是,如果當前位置的 dungeon[i][j] 是?個?較?的正數的話, dp[i][j] 的值可能變成 0 或者負數。也就是最低點數會?于 1 ,那么騎?就會死亡。因此我們求出來的 dp[i][j] 如果?于等于 0 的話,說明此時的最低初始值應該為 1 。處理這種情況僅需讓 dp[i][j] 與 1 取?個最?值即可:dp[i][j] = max(1, dp[i][j])

  1. 初始化:
    可以在最前?加上?個「輔助結點」,幫助我們初始化。使?這種技巧要注意兩個點:
    • 輔助結點??的值要「保證后續填表是正確的」;
    • 「下標的映射關系」。

在本題中,在 dp 表最后?添加??,并且添加?列后,所有的值都先初始化為?窮?,然后讓dp[m][n - 1] = dp[m - 1][n] = 1 即可。

  1. 填表順序:
    根據「狀態轉移?程」,我們需要「從下往上填每??」,「每??從右往左」。

  2. 返回值:
    根據「狀態表?」,我們需要返回 dp[0][0] 的值

🚩代碼實現

class Solution {public int calculateMinimumHP(int[][] d) {// 1. 創建 dp 表// 2. 初始化// 3. 填表// 4. 返回值int m = d.length;int n = d[0].length;int[][] dp = new int[m + 1][n + 1];for(int j = 0; j <= n; j++) {dp[m][j] = Integer.MAX_VALUE;}for(int i = 0; i <= m; i++) {dp[i][n] = Integer.MAX_VALUE;}dp[m][n - 1] = dp[m - 1][n] = 1;for(int i = m - 1; i >= 0; i--) {for(int j = n - 1; j >= 0; j--) {dp[i][j] = Math.min(dp[i][j + 1], dp[i + 1][j]) - d[i][j];dp[i][j] = Math.max(dp[i][j], 1);}}return dp[0][0];}
}

?總結

關于《【算法優選】 動態規劃之路徑問題——貳》就講解到這兒,感謝大家的支持,歡迎各位留言交流以及批評指正,如果文章對您有幫助或者覺得作者寫的還不錯可以點一下關注,點贊,收藏支持一下!

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/214390.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/214390.shtml
英文地址,請注明出處:http://en.pswp.cn/news/214390.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

viple與物理機器人(一):線控模擬

為了檢測viple程序與物理機器人是否能順利連接上 如果能順利連接上&#xff0c;那么&#xff0c;可以通過內建事件從而控制物理機器人的前進、后退、左轉、右轉以及暫停。 如果不能連接上&#xff0c;首先&#xff0c;程序無法控制物理機器人&#xff0c;其次&#xff0c;當vip…

公交站間的距離

&#x1f388; 算法并不一定都是很難的題目&#xff0c;也有很多只是一些代碼技巧&#xff0c;多進行一些算法題目的練習&#xff0c;可以幫助我們開闊解題思路&#xff0c;提升我們的邏輯思維能力&#xff0c;也可以將一些算法思維結合到業務代碼的編寫思考中。簡而言之&#…

我的 CSDN 三周年創作紀念日:2020-12-12

本人大叔一枚&#xff0c;自1992年接觸電腦&#xff0c;持續了30年的業余電腦發燒愛好者&#xff0c;2022年CSDN博客之星Top58&#xff0c;阿里云社區“乘風者計劃”專家博主。自某不知名財校畢業后進入國有大行工作至今&#xff0c;先后任職于某分行信息科技部、電子銀行部、金…

C語言面試之旅:掌握基礎,探索深度(面試實戰之單片機并行存儲器擴展)

引言 在嵌入式系統和微控制器等應用中&#xff0c;存儲器是至關重要的組成部分。單片機通常具有有限的內核存儲器和外部接口&#xff0c;因此擴展存儲器成為許多應用的必要步驟。本文將探討單片機并行存儲器擴展的各個方面。 1、單片機并行擴展總線 并行存儲器擴展是一種將…

《第一行代碼:Android》第三版7.4SQLite數據庫存儲

布局文件略過&#xff0c;就是五個按鈕&#xff0c;點擊按鈕執行對應的功能。 Android 專門提供了一個SQLiteOpenHelper幫助類來對數據庫進行創建和升級。 自己創建一個類繼承自SQLiteOpenHelper,重新寫onCreate()方法和onUpgrade()方法&#xff0c;分別對應創建數據庫和升級…

扔掉xshell,基于 QT 實現一個串口命令行工具(帶源碼)

背景 xshell 帶有支持串口的命令行能力&#xff0c; 可以方便的和下位機用命令進行交互&#xff0c;如下圖所示&#xff1a; msh > msh > msh >version\ | / - RT - Thread Operating System/ | \ 3.1.3 build Nov 7 20232006 - 2019 Copyright by rt-thre…

this.$emit(‘update:isVisible‘, false)作用

這個寫是不是很新穎&#xff0c;傳父組件傳值&#xff01;這是什么鬼。。。 假設你有以下邏輯業務。在A頁面彈出一個組件B&#xff0c;A組件里面使用B組件&#xff0c;是否展示B組件你使用的是baselineShow變量控制&#xff01; <BaselineData :isVisible.sync"basel…

如何在Word中簡潔地插入代碼

如何在Word中簡潔地插入代碼 背景&#xff1a; ? 最近在一寫一些論文或者報告的時候&#xff0c;需要將源代碼放在論文的最后&#xff0c;有一個很頭疼的問題&#xff0c;如果直接把代碼從編輯器復制到word中&#xff0c;就變成了下面這個樣子&#xff1a; 這有點丑陋啊&…

Qt簡介、C++工程文件分離、創建Qt工程、Qt的幫助文檔

QT 簡介 core&#xff1a;核心模塊&#xff0c;非圖形的接口類&#xff0c;為其它模塊提供支持 gui&#xff1a;圖形用戶接口&#xff0c;qt5之前 widgets&#xff1a;圖形界面相關的類模塊 qt5之后的 database&#xff1a;數據庫模塊 network&#xff1a;網絡模塊 QT 特性 開…

Linux系統的各項命令

文章目錄 Linux系統的目錄結構Linux路徑的描述方式Linux命令入門**什么是命令、命令行**Linux命令基礎格式 ls命令入門HOME目錄和工作目錄ls命令的參數和選項ls命令的 -a選項ls命令的 -l選項ls命令選項的組合使用ls選項和參數的組合使用ls命令的 -h選項 目錄切換相關命令&#…

多線程案例-阻塞隊列

阻塞隊列是什么 阻塞隊列是一種特殊的隊列.也遵循"先進先出"的原則 阻塞隊列能是一種線程安全的數據結構,并且具有以下特性: 當隊列滿的時候,繼續入隊列就會阻塞,直到有其他線程從隊列中取走元素. 當隊列空的時候,繼續出隊列也會阻塞,直到有其他線程往隊列中插入元素…

這七款網工在線畫拓撲工具,絕了!

你們好&#xff0c;我的網工朋友。 畫拓撲圖&#xff0c;絕對是網絡工程師的基操。 上次給你來了篇手把手教你繪制拓撲圖的好文&#xff0c;還沒看過的先去看啊&#xff1a;《網絡拓撲圖怎么畫最好&#xff1f;》。 關于畫拓撲的工具&#xff0c;那就多了&#xff0c;直接用…

數據結構與算法-D8D9隊列實現及應用

隊列&#xff1a;限制在兩端進行插入和刪除的線性表 允許進行存入操作的一端為“隊尾” 允許進行刪除操作的一端為“隊頭” 順序隊列 注意&#xff1a;front指向隊頭元素的位置 rear指向隊尾元素的下一個位置 實現循環隊列&#xff1a;(rear1)%N取余&#xff0c;為了區分空…

Connection refused: no further information

解決目錄 一、報錯信息二、解決方法 一、報錯信息 二、解決方法 1、報錯原因是開啟了代理&#xff0c;像AS是絕對不能開代理的。 2、設置為No proxy&#xff0c;然后Apply再選擇OK&#xff0c;重新同步。 要遠離消耗你的人和事&#xff0c;不要花費任何情緒或者精力在他們身…

unity Pc獲取本機Mac地址

1.此方法只能獲取眾多Mac中的一個 private static string GetMacAddress(){string physicalAddress "";NetworkInterface[] nice NetworkInterface.GetAllNetworkInterfaces();foreach (NetworkInterface adaper in nice){Debug.Log(adaper.Description);if (adape…

Linux網絡——高級IO

目錄 一.五種IO模型 1.阻塞式IO 2.非阻塞式IO 3.信號驅動IO 4.多路轉接IO&#xff1a; 5.異步IO 二.同步通信 vs 異步通信 三.設置非阻塞IO 1.阻塞 vs 非阻塞 2.非阻塞IO 3.實現函數SetNoBlock 四.I/O多路轉接之select 1.初識select 2.select函數原型 3.socket就緒…

UEFI下Windows10和Ubuntu22.04雙系統安裝圖解

目錄 簡介制作U盤啟動盤并從U盤啟動電腦安裝系統安裝Windows系統安裝Ubuntu 附錄雙系統時間不一致 簡介 傳統 Legacy BIOS主板下的操作系統安裝可參考本人博客 U盤系統盤制作與系統安裝&#xff08;詳細圖解&#xff09; &#xff0c;本文介紹UEFI主板下的雙系統安裝&#xff…

手把手教你在GPU T4卡上安裝硬解環境+編譯硬解的ffmpeg

系列文章目錄 文章目錄 系列文章目錄前言一、NVDIA環境軟件安裝二、FFMPEG編譯過程總結前言 通常開發流媒體服務,經常需要ffmpeg支持硬解解碼功能,即常見的GPU解碼,如cuda解碼等。下面主要講解在全新的環境中怎么安裝nvidia的環境與編譯ffmpeg的過程。 運行環境Centos7.5 G…

解決 Element-ui中 表格(Table)使用 v-if 條件切換后,表格的列的篩選不顯示了

解決方法 在每個需要使用 v-if 或 v-else 的 el-table-column 上增加 key 作為唯一標識&#xff0c;這樣渲染的時候就不會因為復用原則導致列數據混亂了。關于key值&#xff0c;一般習慣使用字段名&#xff0c;也可隨機生成一個值&#xff0c;只要具有唯一性就可以。

如何快速上手不熟悉的庫

首先需要一個編輯器vscode或者pycharm 然后&#xff0c;不要傻乎乎的自己急著去看代碼。 先看有沒有文檔和使用手冊&#xff0c;一般都有一個quick_start.md文件或者其他的.md文件。 然后&#xff0c;還是不急著看代碼&#xff0c;先看代碼的注釋。 比如我現在要從這里找到…