本篇主要介绍动态规划解题思路
动态规划主要是解一些递归问题,也就是将递归写成非递归方式,因为编辑器无法正确对待递归,递归方法会导致很多计算结果被重复计算,比如菲波那切数列。
所以动态规划的解题思路也就是
比如菲波那切数列
F(n) = F(n-1) + F(n-2)
使用两个中间变量存储之前的计算结果,就改写成了非递归方式实现,也就是动态规划。
leetcode 动态规划题
https://leetcode-cn.com/explore/interview/card/top-interview-questions-easy/23/dynamic-programming/
一. 爬楼梯
假设你正在爬楼梯。需要 n 步你才能到达楼顶。
每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢?
按照解题方法,先列出递归方程
设定dp[i] 表示走到某个台阶的方法,那么就有递归方程
dp[1] = 1;dp[2] = 2(一次走一步/一次走二步)
dp[3] = dp[1] + dp[2] (从dp[1]中方法中,走2步上来,或者从dp[2]种方法中走1步上来)
dp[n] = dp[n-2] + dp[n-1]
写成非递归方法
public static int climbStairs(int n) {
if(n < 1) {
return 0;
}
if (n == 1) {
return 1;
}
if (n == 2) {
return 2;
}
int dp[] = new int [] {1,2};
int result = dp[0] + dp[1];
for (int i = 3; i <= n; i++) {
result = dp[0] + dp[1];
dp[0] = dp[1];
dp[1] = result;
}
return result;
}
二. 最大子序列
给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。
代码如下
public static int maxSubArray(int[] nums) {
if (nums.length == 0) {
return 0;
}
if (nums.length == 1) {
return nums[0];
}
int result = Math.max(nums[0]+nums[1],nums[1]);
int dp = result;
int max = Math.max(result, nums[0]);
for (int i = 2; i< nums.length; i++) {
result = Math.max(dp+nums[i],nums[i]);
dp = result;
// 如果比历史的大,就替换
if (result > max) {
max = result;
}
}
return max;
}
三. 打家劫舍
你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。
给定一个代表每个房屋存放金额的非负整数数组,计算你在不触动警报装置的情况下,能够偷窃到的最高金额
如 [2,7,9,3,1]
设定dp[i] 是偷i个房屋,所能偷窃到的最高金额,dp[1] = 2, dp[2] = 7, dp[3] = 2+9,Math.max(dp[1] + nums[3], dp[2])
递归方程
dp[i] = max(dp[i-2] + nums[i], dp[i-1])
非递归方法实现
public static int rob(int[] nums) {
if (nums.length ==0) {
return 0;
}
if (nums.length == 1) {
return nums[0];
}
if (nums.length ==2) {
return Math.max(nums[0],nums[1]);
}
int dp[] = new int[] {nums[0], Math.max(nums[0], nums[1])};
int result=0;
for (int i = 2; i < nums.length; i++) {
result = Math.max(dp[0] + nums[i], dp[1]);
dp[0] = dp[1];
dp[1] = result;
}
return result;
}