很多讀者對動態(tài)規(guī)劃問題的 base case、備忘錄初始值等問題存在疑問。
本文就專門講一講這類問題,順便聊一聊怎么通過題目的蛛絲馬跡揣測出題人的小心思,輔助我們解題。
看下力扣第 931 題「下降路徑最小和」,輸入為一個n * n
的二維數(shù)組matrix
,請你計算從第一行落到最后一行,經(jīng)過的路徑和最小為多少。
函數(shù)簽名如下:
int minFallingPathSum(int[][] matrix);
就是說你可以站在matrix
的第一行的任意一個元素,需要下降到最后一行。
每次下降,可以向下、向左下、向右下三個方向移動一格。也就是說,可以從matrix[i][j]
降到matrix[i+1][j]
或matrix[i+1][j-1]
或matrix[i+1][j+1]
三個位置。
請你計算下降的「最小路徑和」,比如說題目給了一個例子:
今天這道題也是類似的,不算是困難的題目,所以 我們借這道題來講講 base case 的返回值、備忘錄的初始值、索引越界情況的返回值如何確定 。
不過還是要通過 動態(tài)規(guī)劃的標準套路介紹一下這道題的解題思路,首先我們可以定義一個dp
數(shù)組:
int dp(int[][] matrix, int i, int j);
這個dp
函數(shù)的含義如下:
從第一行(matrix[0][..]
)向下落,落到位置matrix[i][j]
的最小路徑和為dp(matrix, i, j)
。
根據(jù)這個定義,我們可以把主函數(shù)的邏輯寫出來:
public int minFallingPathSum(int[][] matrix) {
int n = matrix.length;
int res = Integer.MAX_VALUE;
// 終點可能在最后一行的任意一列
for (int j = 0; j < n; j++) {
res = Math.min(res, dp(matrix, n - 1, j));
}
return res;
}
因為我們可能落到最后一行的任意一列,所以要窮舉一下,看看落到哪一列才能得到最小的路徑和。
接下來看看dp
函數(shù)如何實現(xiàn)。
對于matrix[i][j]
,只有可能從matrix[i-1][j], matrix[i-1][j-1], matrix[i-1][j+1]
這三個位置轉(zhuǎn)移過來。
那么,只要知道到達(i-1, j), (i-1, j-1), (i-1, j+1)
這三個位置的最小路徑和,加上matrix[i][j]
的值,就能夠計算出來到達位置(i, j)
的最小路徑和 :
int dp(int[][] matrix, int i, int j) {
// 非法索引檢查
if (i < 0 || j < 0 ||
i >= matrix.length ||
j >= matrix[0].length) {
// 返回一個特殊值
return 99999;
}
// base case
if (i == 0) {
return matrix[i][j];
}
// 狀態(tài)轉(zhuǎn)移
return matrix[i][j] + min(
dp(matrix, i - 1, j),
dp(matrix, i - 1, j - 1),
dp(matrix, i - 1, j + 1)
);
}
int min(int a, int b, int c) {
return Math.min(a, Math.min(b, c));
}
當然,上述代碼是暴力窮舉解法,我們可以用備忘錄的方法消除重疊子問題,完整代碼如下:
public int minFallingPathSum(int[][] matrix) {
int n = matrix.length;
int res = Integer.MAX_VALUE;
// 備忘錄里的值初始化為 66666
memo = new int[n][n];
for (int i = 0; i < n; i++) {
Arrays.fill(memo[i], 66666);
}
// 終點可能在 matrix[n-1] 的任意一列
for (int j = 0; j < n; j++) {
res = Math.min(res, dp(matrix, n - 1, j));
}
return res;
}
// 備忘錄
int[][] memo;
int dp(int[][] matrix, int i, int j) {
// 1、索引合法性檢查
if (i < 0 || j < 0 ||
i >= matrix.length ||
j >= matrix[0].length) {
return 99999;
}
// 2、base case
if (i == 0) {
return matrix[0][j];
}
// 3、查找備忘錄,防止重復計算
if (memo[i][j] != 66666) {
return memo[i][j];
}
// 進行狀態(tài)轉(zhuǎn)移
memo[i][j] = matrix[i][j] + min(
dp(matrix, i - 1, j),
dp(matrix, i - 1, j - 1),
dp(matrix, i - 1, j + 1)
);
return memo[i][j];
}
int min(int a, int b, int c) {
return Math.min(a, Math.min(b, c));
}
如果看過我們公眾號之前的動態(tài)規(guī)劃系列文章,這個解題思路應該是非常容易理解的。
那么本文對于這個dp
函數(shù)仔細探討三個問題 :
1、對于索引的合法性檢測,返回值為什么是 99999?其他的值行不行?
2、base case 為什么是i == 0
?
3、備忘錄memo
的初始值為什么是 66666?其他值行不行?
首先,說說 base case 為什么是i == 0
,返回值為什么是matrix[0][j]
,這是根據(jù)dp
函數(shù)的定義所決定的 。
回顧我們的dp
函數(shù)定義:
從第一行(matrix[0][..]
)向下落,落到位置matrix[i][j]
的最小路徑和為dp(matrix, i, j)
。
根據(jù)這個定義,我們就是從matrix[0][j]
開始下落。那如果我們想落到的目的地就是i == 0
,所需的路徑和當然就是matrix[0][j]
唄。
再說說備忘錄memo
的初始值為什么是 66666,這是由題目給出的數(shù)據(jù)范圍決定的 。
備忘錄memo
數(shù)組的作用是什么?
就是防止重復計算,將dp(matrix, i, j)
的計算結(jié)果存進memo[i][j]
,遇到重復計算可以直接返回。
那么,我們必須要知道memo[i][j]
到底存儲計算結(jié)果沒有,對吧?如果存結(jié)果了,就直接返回;沒存,就去遞歸計算。
所以,memo
的初始值一定得是特殊值,和合法的答案有所區(qū)分。
我們回過頭看看題目給出的數(shù)據(jù)范圍:
matrix
是n * n
的二維數(shù)組,其中1 <= n <= 100
;對于二維數(shù)組中的元素,有-100 <= matrix[i][j] <= 100
。
假設matrix
的大小是 100 x 100,所有元素都是 100,那么從第一行往下落,得到的路徑和就是 100 x 100 = 10000,也就是最大的合法答案。
類似的,依然假設matrix
的大小是 100 x 100,所有元素是 -100,那么從第一行往下落,就得到了最小的合法答案 -100 x 100 = -10000。
也就是說,這個問題的合法結(jié)果會落在區(qū)間[-10000, 10000]
中。
所以,我們memo
的初始值就要避開區(qū)間[-10000, 10000]
,換句話說,memo
的初始值只要在區(qū)間(-inf, -10001] U [10001, +inf)
中就可以。
最后,說說對于不合法的索引,返回值應該如何確定,這需要根據(jù)我們狀態(tài)轉(zhuǎn)移方程的邏輯確定 。
對于這道題,狀態(tài)轉(zhuǎn)移的基本邏輯如下:
int dp(int[][] matrix, int i, int j) {
return matrix[i][j] + min(
dp(matrix, i - 1, j),
dp(matrix, i - 1, j - 1),
dp(matrix, i - 1, j + 1)
);
}
顯然,i - 1, j - 1, j + 1
這幾個運算可能會造成索引越界,對于索引越界的dp
函數(shù),應該返回一個不可能被取到的值。
因為我們調(diào)用的是min
函數(shù),最終返回的值是最小值,所以對于不合法的索引,只要dp
函數(shù)返回一個永遠不會被取到的最大值即可。
剛才說了,合法答案的區(qū)間是[-10000, 10000]
,所以我們的返回值只要大于 10000 就相當于一個永不會取到的最大值。
換句話說,只要返回區(qū)間[10001, +inf)
中的一個值,就能保證不會被取到。
至此,我們就把動態(tài)規(guī)劃相關的三個細節(jié)問題舉例說明了。
拓展延伸一下,建議大家做題時,除了題意本身,一定不要忽視題目給定的其他信息 。
本文舉的例子,測試用例數(shù)據(jù)范圍可以確定「什么是特殊值」,從而幫助我們將思路轉(zhuǎn)化成代碼。
除此之外,數(shù)據(jù)范圍還可以幫我們估算算法的時間/空間復雜度。
比如說,有的算法題,你只想到一個暴力求解思路,時間復雜度比較高。如果發(fā)現(xiàn)題目給定的數(shù)據(jù)量比較大,那么肯定可以說明這個求解思路有問題或者存在優(yōu)化的空間。
除了數(shù)據(jù)范圍,有時候題目還會限制我們算法的時間復雜度,這種信息其實也暗示著一些東西。
比如要求我們的算法復雜度是O(NlogN)
,你想想怎么才能搞出一個對數(shù)級別的復雜度呢?
肯定得用到 二分搜索或者二叉樹相關的數(shù)據(jù)結(jié)構(gòu),比如TreeMap
,PriorityQueue
之類的對吧。
再比如,有時候題目要求你的算法時間復雜度是O(MN)
,這可以聯(lián)想到什么?
可以大膽猜測,常規(guī)解法是用 [回溯算法暴力窮舉,但是更好的解法是動態(tài)規(guī)劃,而且是一個二維動態(tài)規(guī)劃,需要一個M * N
的二維dp
數(shù)組,所以產(chǎn)生了這樣一個時間復雜度。
如果你早就胸有成竹了,那就當我沒說,畢竟猜測也不一定準確;但如果你本來就沒啥解題思路,那有了這些推測之后,最起碼可以給你的思路一些方向吧?
總之,多動腦筋,不放過任何蛛絲馬跡,你不成為刷題小能手才怪。
-
Base
+關注
關注
0文章
11瀏覽量
8719 -
數(shù)組
+關注
關注
1文章
417瀏覽量
25978 -
動態(tài)規(guī)劃算法
+關注
關注
0文章
6瀏覽量
1636
發(fā)布評論請先 登錄
相關推薦
評論