动态规划 70.climbing Stairs
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的更多相关文章
- 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 ...
- Leetcode之70. Climbing Stairs Easy
Leetcode 70 Climbing Stairs Easy https://leetcode.com/problems/climbing-stairs/ You are climbing a s ...
- 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 ...
- Leetcode#70. Climbing Stairs(爬楼梯)
题目描述 假设你正在爬楼梯.需要 n 阶你才能到达楼顶. 每次你可以爬 1 或 2 个台阶.你有多少种不同的方法可以爬到楼顶呢? 注意:给定 n 是一个正整数. 示例 1: 输入: 2 输出: 2 解 ...
- 377. Combination Sum IV 70. Climbing Stairs
back function (return number) remember the structure class Solution { int res = 0; //List<List< ...
- 刷题70. Climbing Stairs
一.题目说明 题目70. Climbing Stairs,爬台阶(楼梯),一次可以爬1.2个台阶,n层的台阶有几种爬法.难度是Easy! 二.我的解答 类似的题目做过,问题就变得非常简单.首先用递归方 ...
- LeetCode练题——70. Climbing Stairs
1.题目 70. Climbing Stairs——Easy You are climbing a stair case. It takes n steps to reach to the top. ...
- [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 ...
- [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 ...
随机推荐
- C++——STL之vector, list, deque容器对比与常用函数
STL 三种顺序容器的特性对比: vector 可变数组,内存空间是连续的,容量不会进行缩减.支持高效随机存取,即支持[]和at()操作.尾部插入删除效率高,其他位置插删效率较低: list 双向链表 ...
- 2014蓝桥杯B组初赛试题《六角填数》
题目描述: 如图[1.png]所示六角形中,填入1~12的数字. 使得每条直线上的数字之和都相同. 图中,已经替你填好了3个数字,请你计算星号位置所代表的数字是多少? 请通过浏览器提交 ...
- IDEA03 连接数据库、自动生成实体类
1 版本说明 JDK:1.8 MAVEN:3.5 SpringBoot:2.0.4 IDEA:旗舰版207.2 MySQL:5.5 2 利用IDEA连接数据库 说明:本案例以MySQL为例 2.1 打 ...
- zlib编程
一.简介 zlib是提供数据压缩用的函式库,使用DEFLATE算法,最初是为libpng函式库所写的,后来普遍为许多软件所使用,今天,zlib是一种事实上的业界标准. 二.基本信息 数据头(hea ...
- asp.net web 自定义控件
0.调用代码 protected override void Page_Load(object sender, EventArgs e) { //给基类服务接口复制,可不付 if (IsPostBac ...
- html5操作类名API——classlist
tagNode.classList.add('123'); // 添加类 tagNode.classList.remove('bbb'); // 删除类 tagNode.classList.toggl ...
- 2.8.2 并发下的ArrayList,以及源码分析
package 第二章.并发下的ArrayList; import java.util.ArrayList;import java.util.List; /** * Created by zzq on ...
- java 异常捕获与异常处理
try{会产生异常的代码 }catch(出现异常的类型 e){ 异常出现后处理的方法 } 一旦异常出现又没有异常处理,程序就会中断. public static void main(String[] ...
- windows下go调用内存dll
有时候我们希望将dll嵌入到程序内部,以提高程序的安全性,这里我写的一个开源memorydll模块. 首先 go get github.com/nkbai/go-memorydll 然后在需要的时候 ...
- 严选 Android 路由框架优化(上篇)
0 背景 早前严选 Android 工程,使用原生 Intent 方式做页面跳转,为规范参数传递,做了编码规范,使用静态方法的方式唤起 Activity public static void star ...