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. c++ 装饰模式(decorate)

    装饰模式:动态地给一个对象添加一些额外的职责.就增加功能来说,装饰模式相比生成子类 更为灵活.有时我们希望给某个对象而不是整个类添加一些功能.比如有一个手机,允许你为手机添加特性,比如增加挂件.屏幕贴 ...

  2. 怎样使用Mock Server

    转载自:http://www.cnblogs.com/111testing/p/6091460.html 怎样使用Mock Server   一,去这里https://github.com/dream ...

  3. 2.Hive的几种常见的数据导入方式

    好久没写Hive的那些事了,今天开始写点吧.今天的话题是总结Hive的几种常见的数据导入方式,我总结为四种:(1).从本地文件系统中导入数据到Hive表:(2).从HDFS上导入数据到Hive表:(3 ...

  4. 10.IN 操作符

    IN 操作符 IN 操作符允许我们在 WHERE 子句中规定多个值. SQL IN 语法 SELECT column_name(s) FROM table_name WHERE column_name ...

  5. 在Windows 8上安装SQL Server2012

    SQL Server 2012 的安装方法跟2008差不多,基本上都是点击下一步,不过在安装的时候可能会进度条一直停留在“正在启动操作系统功能”NetFx3””处不动,出现这个问题的原因是在Windo ...

  6. Mysql CURD复习(数据库、表、数据)

    ###############################数据库的CURD:C: create database if not exists tp5_test default charset ut ...

  7. 现代C++学习笔记之二入门篇1

    现代 C++ 强调: 基于堆栈的范围,而非堆或静态全局范围. 自动类型推理,而非显式类型名称. 智能指针而不是原始指针. std::string 和 std::wstring 类型(请参见 <s ...

  8. smartUpload上传下载

    上传 file_upload_smart_form.jsp文件代码 <%@ page contentType="text/html;charset=gb2312" langu ...

  9. delphi 指针 认识

    delphi 指针分为类型指针和无类型指针: 类型指针分为PChar.PInteger.PString等. 无类型指针Pointer. PPChar/PP...为指针的指针 @和Addr一样,为获取变 ...

  10. MVC区域路由配置