前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >初级算法-动态规划

初级算法-动态规划

作者头像
方丈的寺院
发布2019-08-05 17:30:29
3090
发布2019-08-05 17:30:29
举报
文章被收录于专栏:方丈的寺院方丈的寺院

摘要

本篇主要介绍动态规划解题思路

概括

动态规划主要是解一些递归问题,也就是将递归写成非递归方式,因为编辑器无法正确对待递归,递归方法会导致很多计算结果被重复计算,比如菲波那切数列。

所以动态规划的解题思路也就是

  1. 列出递归表达式
  2. 将递归方法写成非递归方式

比如菲波那切数列 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] 表示走到某个台阶的方法,那么就有递归方程

代码语言:javascript
复制
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]

写成非递归方法

代码语言:javascript
复制
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 ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

  1. 列出递归方程 以 nums[-2,1,-3,4,-1,2,1,-5,4] 为例, dp[i]表示遍历到第i个数时,当前最大连续子数组和 dp[1] = -2, dp[2] = Math.max(dp[1] + nums[2],nums[2]),因为要求是连续子数组,所以nums[i]必须接上, 然后再比较历史最大的的连续子数组和这次的最大值

代码如下

代码语言:javascript
复制
 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]) 递归方程

代码语言:javascript
复制
dp[i] = max(dp[i-2] + nums[i], dp[i-1])

非递归方法实现

代码语言:javascript
复制
 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;
    }
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-07-16,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 方丈的寺院 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 摘要
  • 概括
  • 常见的题
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档