动态规划 - 198. 打家劫舍(C#和C实现)

动态规划 - 198. 打家劫舍(C#和C实现)

题目描述

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

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

示例 1:

输入: [1,2,3,1]
输出: 4
解释: 偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。
     偷窃到的最高金额 = 1 + 3 = 4

示例 2:

输入: [2,7,9,3,1]
输出: 12
解释: 偷窃 1 号房屋 (金额 = 2), 偷窃 3 号房屋 (金额 = 9),接着偷窃 5 号房屋 (金额 = 1)。
     偷窃到的最高金额 = 2 + 9 + 1 = 12

解题思路

辅助理解:

假设我们要偷一排房屋,每间房屋都有一定的金额。如果我们只能偷一间房屋,那么我们当然要偷金额最高的房屋。如果我们可以偷多间房屋,那么我们可以选择偷金额最高的两间房屋,或者偷金额最高的三间房屋,以此类推。

动态规划
  1. 定义状态:dp[i] 表示偷窃到第 i 间房屋时能够获得的最高金额。
  2. 状态转移方程: dp[i] = max(dp[i-1], dp[i-2] + nums[i]),即偷窃到第 i 间房屋的最高金额等于偷窃到第 i-1 间房屋的最高金额和偷窃到第 i-2 间房屋的最高金额加上当前房屋的金额的最大值。
  3. 初始状态: dp[0] = nums[0]dp[1] = max(nums[0], nums[1])
  4. 遍历顺序: 从小到大遍历,计算每一间房屋的最高金额。
特殊案例
  • 如果数组长度为 0,则返回 0。
  • 如果数组长度为 1,则返回数组第一个元素的值。

C#代码实现

public int Rob(int[] nums) {
    int n = nums.Length;

    if (n == 0) {
        return 0;
    }

    if (n == 1) {
        return nums[0];
    }

    int[] dp = new int[n];
    dp[0] = nums[0];
    dp[1] = Math.Max(nums[0], nums[1]);

    for (int i = 2; i < n; i++) {
        dp[i] = Math.Max(dp[i - 1], dp[i - 2] + nums[i]);
    }

    return dp[n - 1];
}

C代码实现

int rob(int* nums, int numsSize) {
    if (numsSize == 0) {
        return 0;
    }

    if (numsSize == 1) {
        return nums[0];
    }

    int* dp = (int*)malloc(sizeof(int) * numsSize);
    dp[0] = nums[0];
    dp[1] = fmax(nums[0], nums[1]);

    for (int i = 2; i < numsSize; i++) {
        dp[i] = fmax(dp[i - 1], dp[i - 2] + nums[i]);
    }

    int result = dp[numsSize - 1];
    free(dp);

    return result;
}

时间复杂度和空间复杂度

  • 时间复杂度:O(n),其中 n 是房屋的数量。需要计算每一间房屋的最高金额。
  • 空间复杂度:O(n)。使用了一个大小为 n 的数组来保存中间结果。

参与点评

读者朋友们,如果您在阅读过程中,对文章的质量、易理解性有任何建议,欢迎在评论区指出,我会认真改进。

  • 15
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AmHardy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值