20230619周一算法题【打家劫舍系列】

198. 打家劫舍

你是一个专业的法外狂徒,你叫张三,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

思路溯源

方法一:动态规划:

第k间房偷的话,最高金额是前k-2间房的最高金额加上第k间的金额
不偷的话最高是第k-1间房的的最高金额

我们设置一个dp数组,表示面对第i间房的时候能偷的最大值

  • 初始化状态:
    面对第0间房,你别无选择,能偷的最大值就是第0间房
    面对第一间房,你注定只能偷一间,所以最大值是Math.max(nums[0], nums[1])
    由于考察任意一件房,只需要知道它前两间房的最高金额即可,所以初始状态设置完毕
  • 对任意间(i)房进行考察
    根据

第k间房偷的话,最高金额是前k-2间房的最高金额加上第k间的金额
不偷的话最高是第k-1间房的的最高金额

所以第i间的最大金额就是dp[i - 2] + nums[i]和dp[i - 1]里面取最大值
dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1])

var rob = function(nums) {
    const len = nums.length
    let dp = [nums[0], Math.max(nums[0], nums[1])] //如果长度是1,这里Math.max(nums[0], nums[1])是NaN,不会报错
    if(len === 1 ) return dp[0]
    if(len === 2 ) return dp[1]
    for(let i = 2; i < len; i++){
        dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1])
    }
    return dp[len - 1]
};

由于我们真正需要的只有i-1和i-2两间房的信息,所以这里可以做下优化
pre就是i-2的最大金额
cur就是-1的最大金额
然后不断地将这两个值沿着数组推进更新

var rob = function(nums) {
    const len = nums.length;
    let pre = 0; // i-2的最大金额
    let cur = 0; //i-1 的最大金额

    for (let i = 0; i < len; i++) {
        const next = Math.max(pre + nums[i], cur); //当前房屋的最大金额
        pre = cur; //这两个值向前推进
        cur = next;
    }

    return cur;
};
方法二:直接自首:

在评论区看到有人说他直接return 110 自首。我觉得也应该给分,非常符合核心价值观

var rob = function(nums) {
    return 110;
};

213. 打家劫舍 II

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警 。

给定一个代表每个房屋存放金额的非负整数数组,计算你 在不触动警报装置的情况下 ,今晚能够偷窃到的最高金额。

解法分析:

和法外狂徒一的区别就是把房子围成了一圈,也就是说第一间房和最后一间房能不能偷,他俩现在互相影响了
那咱们有了在上面那个地方的违法经历,现在比较有经验了,我们要考虑如何把这一题分解下,变成和上一题一样的情景

方法一:

既然题目的变化是成为了一个圈,那我们就要把圈拆开
分情况讨论下:
考察第一间房,那么可以分两种情况:假设第一间被偷和假设第一间没被偷,在dp数组的开头和结尾做特殊操作

  • 第一间被偷:
    那么第一间:dp[0] = nums[0]
    第二间一定不能偷,所以:dp[1] = nums[0]
    最后一间只能是不被偷,所以:dp[n] = dp[n-1]
  • 第一间没有被偷
    那么第一件不能被偷:dp[0] = 0
var rob = function(nums) {
    const len = nums.length
    if(len === 1 ) return nums[0]
    if(len === 2 ) return Math.max(...nums)
    //第一间被偷:所以最后一间只能是不被偷,所以dp[n] = dp[n-1]
    let dp = [nums[0], nums[0]]
    for(let i = 2; i < len; i++){
        if(i === len - 1){
          dp[i]  = dp[i - 1]
        } else {
            dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1])
        }
    }
    //第一间没被偷,所以dp[0] = 0
    dp1 = [0, Math.max(0, nums[1])]
    for(let i = 2; i < len; i++){
            dp1[i] = Math.max(dp1[i - 2] + nums[i], dp1[i - 1])
    }

    return Math.max(dp[len - 1], dp1[len - 1 ]) 
};
方法二:

分两种情况:第一间没被偷和最后一间没被偷
第一间没被偷,意味着第2~n间你随便偷
第n间没被偷,意味着第1~n-1间你随便偷

那么为什么这两种情况包含了所有的可能呢?
在所有情景中可以分为第1和第n间房都被偷了(条件不允许,排除)和第1和第n间房至少有一个没被偷两种情况
第1和第n间房至少有一个没被偷两种情况分为:1偷n不偷,n偷1不偷,两个都不偷。
能被这两种情况覆盖:第1间没被偷、第n间没被偷

所以我们就在,这两种情况的最大金额中求最大值即可

var rob = function(nums) {
    const len = nums.length
    if(len === 1 ) return nums[0]
    if(len === 2 ) return Math.max(nums[0], nums[1])
    //1~n-1随便偷 nums.slice(0,len -1)
    //2~n随便偷 nums.slice(1,len) 
    return Math.max(get(nums.slice(1,len)), get(nums.slice(0,len -1))) 

    function get(nums){ //get方法用于在一个可以随便偷的数组中求最大金额
        const len = nums.length
        let dp = [nums[0], Math.max(nums[0], nums[1])]
        if(len === 1 ) return dp[0]
        if(len === 2 ) return dp[1]
        for(let i = 2; i < len; i++){
            dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1])
        }
        return dp[len - 1]
    }
};

337. 打家劫舍 III

张三又又又拓展了新业务,这个地区只有一个入口,我们称之为 root 。

除了 root 之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果 两个直接相连的房子在同一天晚上被打劫 ,房屋将自动报警。

给定二叉树的 root 。返回 在不触动警报的情况下 ,小偷能够盗取的最高金额 。

“聪明的小偷发现这个地方的房屋排列类似于一棵二叉树”,这小偷怕不是程序员失业再就业

解法分析

偷窃的过程循环往复,同样的,需要用到一个迭代的思想

考察任意一个节点x,那么这个节点有偷与不偷两种情况

  • 当前节点咱偷
    那么两个子节点无法偷窃,那么此节点所能偷到的最大金额就是四个孙子的最大金额之和
  • 当前节点咱不偷
    那么两个子节点相对来说就比较自由,可以偷也可以不偷。
    那么最大金额就是max(左子节点,右子节点)
    • 对于子节点
      能偷的最大金额应当是max(偷此节点,不偷此节点)
方法:

robInternal方法用于求出某个特定节点的最大金额,返回一个数组,数组的第0项表示不偷当前节点可以获得的最大金额,第1 项表示当前节点可以获得的最大金额

var rob = function(root) {
    const result = robInternal(root);
    return Math.max(result[0], result[1]); //0当前节点不偷,1偷
};

function robInternal(root){
    let result = [0, 0]
    if(root != null){
        let left = robInternal(root.left)
        let right = robInternal(root.right)
        result[0] = Math.max(left[0], left[1]) + Math.max(right[0], right[1]) 
        result[1] = left[0] + right[0] + root.val
    }
    return result
}

2560. 打家劫舍 IV

法外狂徒张三由于技艺精湛还有编程思维,所以目前还没被抓到,有机会再跟大家分享打家劫舍IV,敬请期待。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,002评论 6 509
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,777评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,341评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,085评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,110评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,868评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,528评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,422评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,938评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,067评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,199评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,877评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,540评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,079评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,192评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,514评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,190评论 2 357

推荐阅读更多精彩内容