自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

聊聊DP入門之不同路徑

開發(fā) 前端
本文分別給出了深搜,動規(guī),數(shù)論三種方法。深搜當(dāng)然是超時了,順便分析了一下使用深搜的時間復(fù)雜度,就可以看出為什么超時了。然后在給出動規(guī)的方法,依然是使用動規(guī)五部曲,這次我們就要考慮如何正確的初始化了,初始化和遍歷順序其實也很重要!

一個機(jī)器人位于一個 m x n 網(wǎng)格的左上角 (起始點在下圖中標(biāo)記為 “Start” )。

機(jī)器人每次只能向下或者向右移動一步。機(jī)器人試圖達(dá)到網(wǎng)格的右下角(在下圖中標(biāo)記為 “Finish” )。

問總共有多少條不同的路徑?

示例 1:

  • 輸入:m = 3, n = 7
  • 輸出:28

示例 2:

  • 輸入:m = 2, n = 3
  • 輸出:3

解釋:從左上角開始,總共有 3 條路徑可以到達(dá)右下角。

  • 向右 -> 向右 -> 向下
  • 向右 -> 向下 -> 向右
  • 向下 -> 向右 -> 向右

示例 3:

  • 輸入:m = 7, n = 3
  • 輸出:28

示例 4:

  • 輸入:m = 3, n = 3
  • 輸出:6

提示:

  • 1 <= m, n <= 100
  • 題目數(shù)據(jù)保證答案小于等于 2 * 10^9

思路

深搜

這道題目,剛一看最直觀的想法就是用圖論里的深搜,來枚舉出來有多少種路徑。

注意題目中說機(jī)器人每次只能向下或者向右移動一步,那么其實機(jī)器人走過的路徑可以抽象為一顆二叉樹,而葉子節(jié)點就是終點!

如圖舉例:

不同路徑

此時問題就可以轉(zhuǎn)化為求二叉樹葉子節(jié)點的個數(shù),代碼如下:

  1. class Solution { 
  2. private: 
  3.     int dfs(int i, int j, int m, int n) { 
  4.         if (i > m || j > n) return 0; // 越界了 
  5.         if (i == m && j == n) return 1; // 找到一種方法,相當(dāng)于找到了葉子節(jié)點 
  6.         return dfs(i + 1, j, m, n) + dfs(i, j + 1, m, n); 
  7.     } 
  8. public
  9.     int uniquePaths(int m, int n) { 
  10.         return dfs(1, 1, m, n); 
  11.     } 
  12. }; 

大家如果提交了代碼就會發(fā)現(xiàn)超時了!

來分析一下時間復(fù)雜度,這個深搜的算法,其實就是要遍歷整個二叉樹。

這顆樹的深度其實就是m+n-1(深度按從1開始計算)。

那二叉樹的節(jié)點個數(shù)就是 2^(m + n - 1) - 1??梢岳斫馍钏训乃惴ň褪潜闅v了整個滿二叉樹(其實沒有遍歷整個滿二叉樹,只是近似而已)

所以上面深搜代碼的時間復(fù)雜度為,可以看出,這是指數(shù)級別的時間復(fù)雜度,是非常大的。

動態(tài)規(guī)劃

機(jī)器人從(0 , 0) 位置出發(fā),到(m - 1, n - 1)終點。

按照動規(guī)五部曲來分析:

確定dp數(shù)組(dp table)以及下標(biāo)的含義

dp[i][j] :表示從(0 ,0)出發(fā),到(i, j) 有dp[i][j]條不同的路徑。

確定遞推公式

想要求dp[i][j],只能有兩個方向來推導(dǎo)出來,即dp[i - 1][j] 和 dp[i][j - 1]。

此時在回顧一下 dp[i - 1][j] 表示啥,是從(0, 0)的位置到(i - 1, j)有幾條路徑,dp[i][j - 1]同理。

那么很自然,dp[i][j] = dp[i - 1][j] + dp[i][j - 1],因為dp[i][j]只有這兩個方向過來。

dp數(shù)組的初始化

如何初始化呢,首先dp[i][0]一定都是1,因為從(0, 0)的位置到(i, 0)的路徑只有一條,那么dp[0][j]也同理。

所以初始化代碼為:

  1. for (int i = 0; i < m; i++) dp[i][0] = 1; 
  2. for (int j = 0; j < n; j++) dp[0][j] = 1; 

確定遍歷順序

這里要看一下遞歸公式dp[i][j] = dp[i - 1][j] + dp[i][j - 1],dp[i][j]都是從其上方和左方推導(dǎo)而來,那么從左到右一層一層遍歷就可以了。

這樣就可以保證推導(dǎo)dp[i][j]的時候,dp[i - 1][j] 和 dp[i][j - 1]一定是有數(shù)值的。

舉例推導(dǎo)dp數(shù)組

如圖所示:

不同路徑

以上動規(guī)五部曲分析完畢,C++代碼如下:

  1. class Solution { 
  2. public
  3.     int uniquePaths(int m, int n) { 
  4.         vector<vector<int>> dp(m, vector<int>(n, 0)); 
  5.         for (int i = 0; i < m; i++) dp[i][0] = 1; 
  6.         for (int j = 0; j < n; j++) dp[0][j] = 1; 
  7.         for (int i = 1; i < m; i++) { 
  8.             for (int j = 1; j < n; j++) { 
  9.                 dp[i][j] = dp[i - 1][j] + dp[i][j - 1]; 
  10.             } 
  11.         } 
  12.         return dp[m - 1][n - 1]; 
  13.     } 
  14. }; 

時間復(fù)雜度:

空間復(fù)雜度:

其實用一個一維數(shù)組(也可以理解是滾動數(shù)組)就可以了,但是不利于理解,可以優(yōu)化點空間,建議先理解了二維,在理解一維,C++代碼如下:

  1. class Solution { 
  2. public
  3.     int uniquePaths(int m, int n) { 
  4.         vector<int> dp(n); 
  5.         for (int i = 0; i < n; i++) dp[i] = 1; 
  6.         for (int j = 1; j < m; j++) { 
  7.             for (int i = 1; i < n; i++) { 
  8.                 dp[i] += dp[i - 1]; 
  9.             } 
  10.         } 
  11.         return dp[n - 1]; 
  12.     } 
  13. }; 

時間復(fù)雜度:

空間復(fù)雜度:

數(shù)論方法

在這個圖中,可以看出一共m,n的話,無論怎么走,走到終點都需要 m + n - 2 步。

不同路徑

在這m + n - 2 步中,一定有 m - 1 步是要向下走的,不用管什么時候向下走。

那么有幾種走法呢?可以轉(zhuǎn)化為,給你m + n - 2個不同的數(shù),隨便取m - 1個數(shù),有幾種取法。

那么這就是一個組合問題了。

那么答案,如圖所示:

不同路徑

求組合的時候,要防止兩個int相乘溢出! 所以不能把算式的分子都算出來,分母都算出來再做除法。

例如如下代碼是不行的。

  1. class Solution { 
  2. public
  3.     int uniquePaths(int m, int n) { 
  4.         int numerator = 1, denominator = 1; 
  5.         int count = m - 1; 
  6.         int t = m + n - 2; 
  7.         while (count--) numerator *= (t--); // 計算分子,此時分子就會溢出 
  8.         for (int i = 1; i <= m - 1; i++) denominator *= i; // 計算分母 
  9.         return numerator / denominator; 
  10.     } 
  11. }; 

需要在計算分子的時候,不斷除以分母,代碼如下:

  1. class Solution { 
  2. public
  3.     int uniquePaths(int m, int n) { 
  4.         long long numerator = 1; // 分子 
  5.         int denominator = m - 1; // 分母 
  6.         int count = m - 1; 
  7.         int t = m + n - 2; 
  8.         while (count--) { 
  9.             numerator *= (t--); 
  10.             while (denominator != 0 && numerator % denominator == 0) { 
  11.                 numerator /= denominator; 
  12.                 denominator--; 
  13.             } 
  14.         } 
  15.         return numerator; 
  16.     } 
  17. }; 

時間復(fù)雜度:

空間復(fù)雜度:

計算組合問題的代碼還是有難度的,特別是處理溢出的情況!

總結(jié)

本文分別給出了深搜,動規(guī),數(shù)論三種方法。

深搜當(dāng)然是超時了,順便分析了一下使用深搜的時間復(fù)雜度,就可以看出為什么超時了。

然后在給出動規(guī)的方法,依然是使用動規(guī)五部曲,這次我們就要考慮如何正確的初始化了,初始化和遍歷順序其實也很重要!

 

責(zé)任編輯:武曉燕 來源: 代碼隨想錄
相關(guān)推薦

2022-01-10 11:28:55

數(shù)據(jù)結(jié)構(gòu)算法DP入門

2021-09-30 11:55:00

微服務(wù)

2022-01-11 10:01:25

二叉搜索樹數(shù)量

2010-06-10 15:36:23

路由協(xié)議的分類

2021-12-28 07:20:44

斐波那契數(shù)算法數(shù)字

2021-05-07 08:02:53

Sentinel 流量服務(wù)

2023-06-05 12:59:03

2021-09-30 09:58:14

路徑總和二叉樹

2021-09-01 22:58:22

Canvas標(biāo)簽

2021-12-29 11:32:38

數(shù)據(jù)結(jié)構(gòu)算法爬樓梯

2024-09-04 09:18:03

分區(qū)策略

2009-12-31 10:03:58

VPN配置實例

2022-12-28 08:16:16

metric聚合java

2021-07-11 12:12:49

.NETJWTjson

2021-06-08 09:28:12

.Net通知服務(wù)

2020-05-27 08:05:33

MybatisMapper接口

2022-03-02 07:52:13

React類組件函數(shù)式組件

2009-12-21 15:04:02

路由器配置

2020-05-06 22:07:53

UbuntuLinux操作系統(tǒng)

2016-11-28 09:00:10

瀏覽器瀏覽器緩存服務(wù)端
點贊
收藏

51CTO技術(shù)棧公眾號