2018-04-29 20:20:56

House Robber问题是leetcode上经典的系列题,这里对其中的题目做一个讲解。

  • 198. House Robber

问题描述:

问题求解:

本质上就是求解不连续取数的情况下能获得最大价值。可以使用动态规划来解决。

dp[i][0]:第i个数没有取能获得的最高价值

dp[i][1]:第i个数取能获得的最高价值

初始值:dp[-1][0] = 0,dp[-1][1] = -INF

递推关系式:dp[i][0] = max(dp[i - 1][0], dp[i - 1][1])

        dp[i][1] = dp[i - 1] + nums[i]

    public int rob(int[] nums) {
int[][] dp = new int[nums.length + 1][2];
dp[0][0] = 0;
dp[0][1] = Integer.MIN_VALUE;
for (int i = 1; i <= nums.length; i++) {
dp[i][0] = Math.max(dp[i - 1][1], dp[i - 1][0]);
dp[i][1] = dp[i - 1][0] + nums[i - 1];
}
return Math.max(dp[nums.length][0], dp[nums.length][1]);
}

可以对空间复杂度进行优化:

    public int rob(int[] nums) {
int dp0 = 0;
int dp1 = Integer.MIN_VALUE;
for (int i = 1; i <= nums.length; i++) {
int tmp = dp0;
dp0 = Math.max(dp1, dp0);
dp1 = tmp + nums[i - 1];
}
return Math.max(dp0, dp1);
}
  • 213. House Robber II

问题描述:

问题求解:

相较于上一题,本题出现了环路,显然的是如果状态是环路的话是无法运用dp的,因为dp是只考虑目前的情况,无后效性,而环路则意味着后面的状态会影响前面的状态,因此,我们要解决这个问题,首先要做的就是考虑使用某种手段来将环路打开。

对于rob问题,如果第i间屋子没有偷窃的话,那么第i + 1间屋子就可以任意取,也就是对于i + 1间屋子来说第i间屋子和不存在是一样的,因此我们可以在没有被偷窃的屋子上将环路断开。

下面考虑第n间屋子,他有两个状态,一是被偷窃,则第 n + 1间屋子必然没有被偷,则从 n + 1 处断开;二是没有被偷,则直接在 n 处断开。

    public int rob(int[] nums) {
if (nums.length == 1) return nums[0];
return Math.max(helper(nums, 0, nums.length - 2), helper(nums, 1, nums.length - 1));
} private int helper(int[] nums, int l, int r) {
int dp0 = 0;
int dp1 = Integer.MIN_VALUE;
for (int i = l; i <= r; i++) {
int tmp = dp0;
dp0 = Math.max(dp0, dp1);
dp1 = tmp + nums[i];
}
return Math.max(dp0, dp1);
}
  • 337. House Robber III

问题描述:

问题求解:

本题是从数据结构的角度对原问题进行了扩展,将场景变到了二叉树上,限制条件依然是不能连续偷取。

从本质上来说,每个结点依然是两个状态,一是偷取了,二是没有偷取,因此,我们依然可以使用动态规划的方法自低向上的将每个结点的左右儿子结点的状态都得到后,再计算自己的值。

    public int rob(TreeNode root) {
int[] res = helper(root);
return Math.max(res[0], res[1]);
} private int[] helper(TreeNode root) {
if (root == null) return new int[2]; int[] res = new int[2];
int[] L = helper(root.left);
int[] R = helper(root.right); res[0] = Math.max(L[0], L[1]) + Math.max(R[0], R[1]);
res[1] = L[0] + R[0] + root.val;
return res;
}

动态规划-House Robber的更多相关文章

  1. leetcode:House Robber(动态规划dp1)

    You are a professional robber planning to rob houses along a street. Each house has a certain amount ...

  2. 一道简单的动态规划题目——House Robber

    一.题目 House Robber(一道Leetcode上的关于动态规划的简单题目)具体描述如下: There is a professional robber planning to rob hou ...

  3. LeetCode之“动态规划”:House Robber && House Robber II

    House Robber题目链接 House Robber II题目链接 1. House Robber 题目要求: You are a professional robber planning to ...

  4. leetcode-198-House Robber(动态规划)

    题目描述: You are a professional robber planning to rob houses along a street. Each house has a certain ...

  5. Leetcode之动态规划(DP)专题-198. 打家劫舍(House Robber)

    Leetcode之动态规划(DP)专题-198. 打家劫舍(House Robber) 你是一个专业的小偷,计划偷窃沿街的房屋.每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互 ...

  6. Leetcode 198 House Robber 动态规划

    题意是强盗能隔个马抢马,看如何获得的价值最高 动态规划题需要考虑状态,阶段,还有状态转移,这个可以参考<动态规划经典教程>,网上有的下的,里面有大量的经典题目讲解 dp[i]表示到第i匹马 ...

  7. 动态规划 - 213. House Robber II

    URL: https://leetcode.com/problems/house-robber-ii/ You are a professional robber planning to rob ho ...

  8. 动态规划 - 198. House Robber

    URL : https://leetcode.com/problems/house-robber/ You are a professional robber planning to rob hous ...

  9. 213. House Robber II(动态规划)

    You are a professional robber planning to rob houses along a street. Each house has a certain amount ...

随机推荐

  1. etc/fstab

    etc/fstab 就是在开机引导的时候自动挂载到linux的文件系统 设备名称 挂载点 分区的类型 挂载选项 dump选项 fsck选项UUID=ce25cdc7-434f-420b-b3 / ex ...

  2. MAC电脑里的休眠功能在哪里?

    Windows7和Ubuntu里都有睡眠和休眠功能,睡眠一般是指挂起到内存,电脑停止运行,数据都在内存里,只需要给内存供电,恢复时很快:休眠是指挂起到硬盘,电脑可以完全停止供电,恢复时从硬盘读取数据, ...

  3. Oracle HA 之 SERVICE和DRM实战

    第一部分:service实战 --oracle 11gR2中创建service的方法:db console和srvctl两种方法. --db console创建service方法-略 --srvctl ...

  4. 理论实践:循序渐进理解AWR细致入微分析性能报告

    1. AWR 概述 Automatic Workload Repository(AWR) 是10g引入的一个重要组件.在里面存贮着近期一段时间内(默认是7天)数据库活动状态的详细信息. AWR 报告是 ...

  5. Django框架【基础篇】

    Python的WEB框架有Django.Tornado.Flask 等多种,Django相较与其他WEB框架其优势为:大而全,框架本身集成了ORM.模型绑定.模板引擎.缓存.Session等诸多功能. ...

  6. SDL结合QWidget的简单使用说明(2)

    上篇主要讲了针对yv12流数据的渲染,但有时候我们显示视频还要求加一些信息,比如头像,昵称等等.一般的想法是在渲染窗口之上做一个小控件来负责: 但是很遗憾,你会发现你的控件被SDL的渲染完全遮住了,渲 ...

  7. 利用阿里云搭建私有Git服务器

    服务器系统:Centos 6 (查看centos版本命令:lsb_release -a) 客户端系统:Windows 7 一.服务器端安装Git ==通常centos上使用yum源安装的git版本过低 ...

  8. JSP学习(第一课)

    JSP页面组成: 比如: 打开网页,右键查看源代码: 打开网页: 注意: <%!%>里面定义的属性是成员属性,相当于类的属性,方法相当于是全局的方法,相当于是类里面的方法.但是它是不可以进 ...

  9. python学习之路-day10

    一.什么是线程 在传统操作系统中,每个进程有一个地址空间,而且默认就有一个控制线程. 线程顾名思义,就是一条流水线工作的过程,一条流水线必须属于一个车间,一个车间的工作过程是一个进程. 车间负责把资源 ...

  10. 前端基础(JavaScript)

    JavaScript概述 JavaScript的历史 1992年Nombas开发出C-minus-minus(C--)的嵌入式脚本语言(最初绑定在CEnvi软件中).后将其改名ScriptEase.( ...