1. 记忆化搜索 - 自上向下的解决问题:使用vector来保存每次计算的结果,如果下次再碰到同样的需要计算的式子就不需要重复计算了。

2. 动态规划 - 自下向上的解决问题

解法一:自顶向下

解法二:自底向上

class Solution {

private:
vector<int> memo; int calcWays(int n){ if(n==) return ; //一个台阶都没有
if(n==) return ;
//if(n==2) return 2; //有两种解决方法:一次迈一步,迈两次;一次迈两步
if(memo[n] == -)
memo[n] = calcWays(n-) + calcWays(n-);
//在第n-1阶台阶迈一步或者在第n-2阶台阶迈两步
return memo[n];
} public:
int climbStairs(int n) {
//n为台阶数
memo = vector<int>(n+,-); //memo初始化为n+1个-1
return calcWays(n);
}
};
class Solution {

public:
int climbStairs(int n) {
//n为台阶数
vector<int> memo(n+,-); //memo初始化为n+1个-1
memo[] = memo[] = ;
for(int i=;i<=n;i++)
memo[i] = memo[i-] + memo[i-];
return memo[n];
}
};

注意:从2只能移动到3和4;从3只能移动到6和5.

思路:设从位置(i,j)达到底部的最小路径和为MP(i,j);根据约束条件,从位置(i,j)只能达到下一行的(i+1,j)和(i+1,j+1)两个位置;

前面的思路是自顶向下的,如果采用自底向上的求解思路,最后的sum[0]是要的结果。可以申请一个一维数组初始化为三角形数阵底部向量,逐步向上计算更新,空间复杂度为O(n)

class Solution {
public:
int minimumTotal(vector<vector<int>>& triangle) {
int length = triangle.size();
if(length==) return ;
if(length==) return triangle[][];
vector<int> sum = triangle[length-]; //初始化sum为三角形底部的向量 for(int i=length-;i>=;i--){
for(int j=;j<triangle[i].size();j++)
sum[j] = min(triangle[i][j]+sum[j], triangle[i][j]+sum[j+]);
} return sum[];
}
};

这个解法是重做了一遍题想到的,感觉比上一个解法有点麻烦,还容易索引溢出。

class Solution {
public:
int minimumTotal(vector<vector<int>>& triangle) {
if(triangle.empty()) return ;
int row = triangle.size();
int column = triangle[row-].size();
int dp[row][column+];
for(int i=; i<row; i++){
for(int j=; j<column+; j++){
dp[i][j] = INT_MAX;
}
}
dp[][] = triangle[][]; for(int i = ; i < row; i++){
for(int j=; j<=i+; j++){ dp[i][j] = min(dp[i-][j-], dp[i-][j]) + triangle[i][j-];
}
} sort(dp[row-], dp[row-]+column+);
return dp[row-][];
}
};

题目:给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

说明:每次只能向下或者向右移动一步。

示例:

输入:
[[1,3,1],
[1,5,1],
[4,2,1]]
输出: 7
解释: 因为路径 1→3→1→1→1 的总和最小。

思路

以输入为 3*3 的网格为例,其中 m=3,n=3
[1,3,1]
[1,5,1]
[4,2,1]
由于每次只能向下或者向右移动,则每一步结果为当前值+min(上边一步,左边一步),即 dp[i][j] = grid[i][j] + min(dp[i-1][j], dp[i][j-1])

注意:不要忘记dp[0][0]的初始化。

class Solution {
public:
int minPathSum(vector<vector<int>>& grid) {
if(grid.size() == ) return ;
int m = grid.size();
int n = grid[].size();
int dp[m][n]; //m行n列的一个二维数组
dp[][] = grid[][];
//初始化边界
for(int i = ; i<m; i++){
dp[i][] = grid[i][] + dp[i-][]; //最左边一列的值只能是当前格子+上面一个
}
for(int i=;i<n;i++){
dp[][i] = grid[][i] + dp[][i-]; //最上面一行的值只能是当前格子+左边一个
}
for(int i=;i<m;i++){
for(int j=;j<n;j++)
dp[i][j] = grid[i][j] + min(dp[i-][j], dp[i][j-]);
} return dp[m-][n-];
}
};

解法二:又做了一遍,自己想出来了思路

1)设置一维数组 dp[i] 为 当前行存储的到该索引的最小和;

2)初始化dp为grid的第一行,即  dp[0] = grid[0][0]; 然后接下来dp[i] = dp[i-1] + grid[0][i];

3) 状态方程:dp[j] = min(dp[j-1]+grid[i][j], dp[j]+grid[i][j]); //取左边或上面到该索引的最小值

4)dp最左边一列只能从上面走到。

class Solution {
public:
int minPathSum(vector<vector<int>>& grid) {
int n = grid.size();
int m = grid[].size();
int dp[m] = {};
dp[] = grid[][];
for(int i=; i<m;i++){
dp[i] = dp[i-] + grid[][i]; } for(int i=; i<n; i++){
for(int j=; j<m ; j++){
if(j==)
dp[j] = dp[j] + grid[i][j];
else{
dp[j] = min(dp[j-]+grid[i][j], dp[j]+grid[i][j]);
}
}
}
return dp[m-];
}
};

动态规划 70.climbing Stairs的更多相关文章

  1. LN : leetcode 70 Climbing Stairs

    lc 70 Climbing Stairs 70 Climbing Stairs You are climbing a stair case. It takes n steps to reach to ...

  2. Leetcode之70. Climbing Stairs Easy

    Leetcode 70 Climbing Stairs Easy https://leetcode.com/problems/climbing-stairs/ You are climbing a s ...

  3. 42. leetcode 70. Climbing Stairs

    70. Climbing Stairs You are climbing a stair case. It takes n steps to reach to the top. Each time y ...

  4. Leetcode#70. Climbing Stairs(爬楼梯)

    题目描述 假设你正在爬楼梯.需要 n 阶你才能到达楼顶. 每次你可以爬 1 或 2 个台阶.你有多少种不同的方法可以爬到楼顶呢? 注意:给定 n 是一个正整数. 示例 1: 输入: 2 输出: 2 解 ...

  5. 377. Combination Sum IV 70. Climbing Stairs

    back function (return number) remember the structure class Solution { int res = 0; //List<List< ...

  6. 刷题70. Climbing Stairs

    一.题目说明 题目70. Climbing Stairs,爬台阶(楼梯),一次可以爬1.2个台阶,n层的台阶有几种爬法.难度是Easy! 二.我的解答 类似的题目做过,问题就变得非常简单.首先用递归方 ...

  7. LeetCode练题——70. Climbing Stairs

    1.题目 70. Climbing Stairs——Easy You are climbing a stair case. It takes n steps to reach to the top. ...

  8. [LeetCode] 70. Climbing Stairs 爬楼梯问题

    You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...

  9. [LeetCode] 70. Climbing Stairs 爬楼梯

    You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...

随机推荐

  1. MySql 之 FIND_IN_SET 和IN

    CREATE TABLE `test` (   `id` int(8) NOT NULL auto_increment,   `name` varchar(255) NOT NULL,   `list ...

  2. 【转】浏览器中F5和CTRL F5的行为区别

    原文地址:http://www.cnblogs.com/jiji262/p/3410518.html 前言 在印象中,浏览器中的F5和刷新按钮是一样的效果,都是对当前页面进行刷新:Ctrl-F5的行为 ...

  3. Solidity 合约调用合约

    原文地址:https://medium.com/@k3no/making-a-birthday-contract-858fd3f63618 先将datetime合约部署:https://github. ...

  4. integer encoding vs 1-hot (py)

    https://github.com/szilard/benchm-ml/issues/1 glouppe commented on 7 May 2015 Thanks for the benchma ...

  5. Part3_lesson4---协处理器访问指令

    1.什么是协处理器? CP15是协处理器, CP15的作用:系统控制协处理器CP15,它提供了额外的寄存器,这些寄存器用于配置和控制cache,MMU,保护系统,时钟模式,和其他的系统项,比如大小端操 ...

  6. You-need-to-know-css

    半透明边框 背景知识: background-clip <div class="main"> <input id="pb" type=&quo ...

  7. try-catch-finally对返回值的影响

    catch 和 finally 一起使用的常见方式是:在 try 块中获取并使用资源,在 catch 块中处理异常情况,并在 finally 块中释放资源. finally 块用于清理try块分配的任 ...

  8. Android绘图之Matrix

    一.概述 1. 在Android中,如果你用Matrix进行过图像处理,那么一定知道Matrix这个类.Android中的Matrix是一个3 x 3的矩阵,其内容如下 2.Matrix的对图像的处理 ...

  9. 策略(Strategy)模式

    /* * 环境(Context)角色:持有一个Strategy类的引用. * 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现.此角色给出所有的具体策略类所需的接口. * ...

  10. java8之流的基本使用(二)

    概述 流(stream())是java8的一个新特性,主要的作用就是将各种类型的集合转换为流,然后的方便迭代数据用的.例如: //将List类型的集合转换为流 list.stream() 转换为流之后 ...