Algorithm-Pattern
  • Introduction
  • C & C++
    • C语言
      • C/C++编译器
      • 宏的使用
      • 编译过程
      • 指针 & 数组
      • 柔性数组
      • 函数指针 & 回调函数
      • C标准库之stdio
      • C标准库之string
      • C标准库之errno
      • C标准库之stdarg
      • C标准库之regex
    • C++基础语法
      • 自增(++) & 自减(--)
      • c语言到c++
      • 可变模板参数
      • 强制类型转换
      • C/C++类型转换的本质
      • 指针 & 引用
      • const的用法
      • static的用法
      • 重要的关键字(一)
      • 重要的关键字(二)
      • 内存申请和释放
      • 内联函数
      • 函数 & 运算符重载
      • 面向对象之封装
      • 构造函数 & 析构函数
      • 面向对象之继承
      • 面向对象之多态
      • 泛型编程
      • 异常
      • 再谈函数指针
    • C++并发编程
      • C++的锁
      • 并发与多线程
    • C++高级特性
      • 函数对象
      • 移动语义 & 完美转发
      • lambda表达式
      • RTTI技术
      • RAII技术
      • 智能指针
      • 模板的特化
      • C++静态库和动态库
      • 内存溢出和内存泄漏
    • STL基础
      • String
      • array/vector/list
      • deque/priority_queue
      • set/map
      • unordered_set/unordered_map
      • algorithm_1
      • functional
      • allocator
    • C++标准库
      • IO
      • Tuple
      • regex
      • bitset
      • numeric
    • STL深入源码
      • vector内部实现
      • deque内部实现
      • sort函数实现
    • 第三方库
      • JsonCpp
      • ProtoBuf
  • 数据结构
    • 线性表
    • 字符串
    • 栈和队列
    • 二叉树
    • 平衡二叉树
    • 平衡多路搜索树
    • 树结构的延申
    • 图
    • 二进制
    • 散列表
  • 算法基础
    • 排序算法
    • 查找算法
    • 数学问题
    • 并查集
    • 递归算法
    • 附加——主定理
    • Catalan数
  • 算法设计思想
    • 滑动窗口思想
    • BFS/DFS
    • 二分法
    • 回溯法
    • 贪心算法
    • 分治法
    • 动态规划
    • 分支限界算法
    • 有限状态机(FSM)
  • LeetCode系列
    • 死磕二叉树
    • 股票买卖问题
    • 打家劫舍问题
    • 跳跃游戏问题
    • 括号匹配问题
    • 石子游戏问题
    • 子序列问题
    • 数组 & 矩阵
    • 排列 & 组合
  • 经典算法问题
    • 几何问题
    • 区间问题
    • 背包问题
    • 石子堆问题
    • 表达式求值
  • 面试题
    • 数据结构和算法基础
    • 程序设计题
      • 实现双线程交替打印
      • C++实现读写锁
      • 实现阻塞队列
      • 实现环形队列
      • 实现线程池
      • 实现智能指针
      • 实现string类
      • 实现高性能local cache
      • 实现内存池
      • 生产者-消费者模型
      • 设计定时器
    • 经典的算法题
    • C++面试题总结
    • 面试算法题总结
由 GitBook 提供支持
在本页
  • 1、打家劫舍【链接】
  • 2、打家劫舍II【链接】
  • 3、打家劫舍III【链接】
  • 树结构的动态规划拓展

这有帮助吗?

  1. LeetCode系列

打家劫舍问题

上一页股票买卖问题下一页跳跃游戏问题

最后更新于4年前

这有帮助吗?

1、打家劫舍【】

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

解决动态规划问题就是找「状态」和「选择」: 房子的索引就是状态,抢和不抢就是选择。

正向思考,先状态,后选择:dp[i] = x 表示抢劫完第i间房屋,最多能抢到的钱为x。

则对于第i间房屋的选择有:1、第i - 1间房屋没抢,则该房屋可以抢,此时有dp[i] = dp[i - 2] + nums[i];2、第i - 1间房屋抢了,则该房屋不能抢,则有dp[i] = dp[i - 1];其实还有第三种选择,第i - 1间房屋没抢,则该房屋可以抢,此时有dp[i - 1] + nums[i];但是该情况下dp[i - 1]<=dp[i - 2],因此只有两种情况,二者取较大者。

空间优化:当前状态只有i-1和i-2两种状态有关,则用两个变量记录状态即可。

int rob(vector<int>& nums) {
    if(nums.empty())
        return 0;
    int len = nums.size();
    int dp[len + 1];
    dp[0] = 0;
    dp[1] = nums[0];
    for(int i = 1; i < len; ++i){
        dp[i + 1] = max(dp[i - 1] + nums[i], dp[i]);
    }
    return dp[len];
}
int rob(vector<int>& nums) {
    if(nums.empty())
        return 0;
    int dp_0 = 0, dp_1 = nums[0];
    for(int i = 1; i < nums.size(); ++i){
        int tmp = dp_1;
        dp_1 = max(dp_0 + nums[i], dp_1);
        dp_0 = tmp;
    }
    return dp_1;
}

反向思考,先选择,后状态:dp[i] = x 表示从第 i 间房子开始抢劫,最多能抢到的钱为 x。

则对于第i间房屋的选择有:1、第i间房屋不抢,则去抢下一间房屋;2、抢第i间房屋,则要跳过下一间房屋;同样还有第三种选择,第i间房屋不抢,则跳过下一间房屋,这种情况包括在第i+1间房屋的选择中,因此只有两种情况,二者取较大者。

int dp(vector<int> &nums, int start){
    if(start >= nums.size()){
        return 0;
    }
    return max(dp(nums, start + 1), /*不抢,去下家*/
               nums[start] + dp(nums, start + 2) /*抢,跳过下家*/
               );
}
int rob(vector<int>& nums) {
    return dp(nums, 0);
}
int dp(vector<int> &nums, int start, vector<int> &memo){
    if(start >= nums.size())
        return 0;
    if(memo[start] != -1)
        return memo[start];
    int res = max(dp(nums, start + 1, memo), nums[start] 
                  + dp(nums, start + 2, memo));
    memo[start] = res;
    return res;
}
int rob(vector<int>& nums) {
    vector<int> memo(nums.size(), -1);
    return dp(nums, 0, memo);
}
int rob(vector<int>& nums) {
    int n = nums.size();
    int dp[n + 2];
    memset(dp, 0, sizeof dp);
    for (int i = n - 1; i >= 0; i--) {
        dp[i] = max(dp[i + 1], nums[i] + dp[i + 2]);
    }
    return dp[0];
}
int rob(vector<int>& nums) {
    int n = nums.size();
    int dp_1 = 0, dp_2 = 0;
    for (int i = n - 1; i >= 0; i--) {
        int tmp = dp_1;
        dp_1 = max(dp_1, nums[i] + dp_2);
        dp_2 = tmp;
    }
    return dp_1;
}

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都围成一圈,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。给定一个代表每个房屋存放金额的非负整数数组,计算你在不触动警报装置的情况下,能够偷窃到的最高金额。

首尾房间不能同时被抢,那么只可能有三种不同情况:要么都不被抢;要么第一间房子被抢最后一间不抢;要么最后一间房子被抢第一间不抢。

在上次打劫完一条街道之后和一圈房屋后,小偷又发现了一个新的可行窃的地区。这个地区只有一个入口,我们称之为“根”。 除了“根”之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果两个直接相连的房子在同一天晚上被打劫,房屋将自动报警。计算在不触动警报的情况下,小偷一晚能够盗取的最高金额。

树结构的动态规划拓展

2、打家劫舍II【】

3、打家劫舍III【】

✏️
链接
✏️
链接
✏️
链接
Binary Tree Cameras