算法打卡day41

2024-05-05 21:36
文章标签 算法 打卡 day41

本文主要是介绍算法打卡day41,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今日任务:

1)198.打家劫舍

2)213.打家劫舍II

3)337.打家劫舍III

4)复习day16

198.打家劫舍

题目链接:198. 打家劫舍 - 力扣(LeetCode)

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。
给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。示例 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。提示:
0 <= nums.length <= 100
0 <= nums[i] <= 400

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,偷不偷这个房间呢?| LeetCode:198.打家劫舍哔哩哔哩bilibili

思路:

这个问题是一个典型的动态规划问题。我们可以使用动态规划来解决这个问题,其中关键是定义状态和状态转移方程。

状态定义:
定义一个数组 dp,其中 dp[i] 表示偷窃前 i 个房屋所能获取的最高金额。

状态转移方程:
考虑偷窃第 i 个房屋,有两种情况:

  1. 偷窃第 i 个房屋:如果偷窃第 i 个房屋,则不能偷窃第 i-1 个房屋,因此偷窃金额为 nums[i] + dp[i-2]
  2. 不偷窃第 i 个房屋:如果不偷窃第 i 个房屋,则偷窃金额与偷窃前 i-1 个房屋所能获取的最高金额相同,即 dp[i-1]

因此,动态转移方程为:

dp[i] = max(dp[i-1], dp[i-2] + nums[i])

初始状态:

  • 第一个房屋只有一间,因此偷窃它即可,即 dp[0] = nums[0]
  • 第二个房屋偷窃与不偷窃中选择金额较大的方案,即 dp[1] = max(nums[0], nums[1])

返回结果:
最终结果为 dp[-1],表示偷窃前所有房屋所能获取的最高金额。

class Solution:def rob(self, nums: List[int]) -> int:if not nums:  # 如果没有房屋,返回0return 0if len(nums) == 1:  # 如果只有一个房屋,返回其金额return nums[0]# 定义动态规划数组,dp[i] 表示偷窃前 i 个房屋所能获取的最高金额dp = [0] * (len(nums))dp[0] = nums[0]  # 第一个房屋只有一间,偷窃它即可dp[1] = max(nums[0], nums[1])  # 前两个房屋,选择金额较大的偷窃# 动态转移方程:偷窃当前房屋与不偷窃当前房屋的最大值for i in range(2, len(nums)):dp[i] = max(dp[i - 1], dp[i - 2] + nums[i])return dp[-1]

213.打家劫舍II

题目链接:213. 打家劫舍 II - 力扣(LeetCode)

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警 。
给定一个代表每个房屋存放金额的非负整数数组,计算你 在不触动警报装置的情况下 ,能够偷窃到的最高金额。示例 1:
输入:nums = [2,3,2]
输出:3
解释:你不能先偷窃 1 号房屋(金额 = 2),然后偷窃 3 号房屋(金额 = 2), 因为他们是相邻的。示例 2:
输入:nums = [1,2,3,1]
输出:4
解释:你可以先偷窃 1 号房屋(金额 = 1),然后偷窃 3 号房屋(金额 = 3)。偷窃到的最高金额 = 1 + 3 = 4 。示例 3:
输入:nums = [0]
输出:0提示:
1 <= nums.length <= 100
0 <= nums[i] <= 1000

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,房间连成环了那还偷不偷呢?| LeetCode:213.打家劫舍II哔哩哔哩bilibili

思路:

这个问题与之前的类似,但有一个额外的条件:房屋围成一圈。因此,我们需要特殊处理这个情况。

我们可以将这个问题拆分成两个子问题:

  1. 偷窃第一个房屋,不偷窃最后一个房屋。
  2. 偷窃最后一个房屋,不偷窃第一个房屋。

然后,我们可以使用动态规划来解决这两个子问题,最后取两个子问题中的最大值作为结果。

具体的动态规划解决方案如下:

  1. 对于第一个子问题,我们可以使用与之前类似的动态规划方法来解决。假设 dp1[i] 表示偷窃前 i 个房屋(第一个房屋被偷窃)所能获取的最高金额,则有以下状态转移方程:

    dp1[i] = max(dp1[i-1], dp1[i-2] + nums[i]) 其中 dp1[0] = nums[0],dp1[1] = max(nums[0], nums[1])
  2. 对于第二个子问题,同样使用动态规划方法。假设 dp2[i] 表示偷窃前 i 个房屋(最后一个房屋被偷窃)所能获取的最高金额,则有以下状态转移方程:

    dp2[i] = max(dp2[i-1], dp2[i-2] + nums[i]) 其中 dp2[0] = 0dp2[1] = nums[1]

最终,我们可以将两个子问题的最终结果进行比较,取其中较大的值作为最终结果。

class Solution:def rob(self, nums: List[int]) -> int:if not nums:  # 如果没有房屋,返回0return 0if len(nums) == 1:  # 如果只有一个房屋,返回其金额return nums[0]# 计算偷窃第一个房屋,不偷窃最后一个房屋的最高金额res1 = self.rob_range(nums,0,len(nums)-2)# 计算偷窃最后一个房屋,不偷窃第一个房屋的最高金额res2 = self.rob_range(nums,1,len(nums)-1)# 取两个子问题的最大值作为最终结果return max(res1,res2)# 动态规划解决子问题def rob_range(self,nums: List[int], start: int, end: int) -> int:if end == start:return nums[start]# 创建一个数组用于存储每个房屋偷窃时的最高金额dp = [0]*(end-start+1)# 初始化前两个房屋的最高金额dp[0] = nums[start]dp[1] = max(nums[start],nums[start+1])for i in range(2, end - start + 1):# 在偷窃当前房屋和不偷窃当前房屋之间选择金额较大的方案dp[i] = max(dp[i-1],dp[i-2]+nums[start+i])return dp[-1]

337.打家劫舍III

题目链接:337. 打家劫舍 III - 力扣(LeetCode)

在上次打劫完一条街道之后和一圈房屋后,小偷又发现了一个新的可行窃的地区。这个地区只有一个入口,我们称之为“根”。 除了“根”之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果两个直接相连的房子在同一天晚上被打劫,房屋将自动报警。
计算在不触动警报的情况下,小偷一晚能够盗取的最高金额。示例 1:
输入: root = [3,2,3,null,3,null,1]
输出: 7
解释: 小偷一晚能够盗取的最高金额 3 + 3 + 1 = 7示例 2:
输入: root = [3,4,5,1,3,null,1]
输出: 9
解释: 小偷一晚能够盗取的最高金额 4 + 5 = 9提示:
树的节点数在 [1, 104] 范围内
0 <= Node.val <= 104

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,房间连成树了,偷不偷呢?| LeetCode:337.打家劫舍3哔哩哔哩bilibili

思路:

在这个问题中,我们有一棵二叉树,表示小偷可以盗取的房屋。每个节点表示一个房屋,节点的值表示房屋内的现金。树中的相邻节点表示相邻的房屋,如果相邻的房屋在同一晚上被打劫,系统会自动报警。

我们的目标是计算在不触发警报的情况下,小偷一晚能够盗取的最高金额。

解决这个问题的一种方法是使用动态规划。我们可以定义一个递归函数rob(root),其中 root 是当前节点。对于每个节点,我们有两种选择:

  1. 盗取当前节点和其孙子节点的价值,然后考虑其孙子节点的左右孩子。
  2. 不盗取当前节点,而是考虑其两个子节点。

我们可以通过递归地计算这两种选择来获得最大金额。但是,这样的方法可能会导致大量的重复计算。为了避免这种情况,我们可以使用记忆化搜索或动态规划来优化递归解法。在动态规划方法中,我们可以定义一个状态数组 dp,其中 dp[node] 表示盗取以 node 为根节点的子树所能获得的最大金额。然后,我们可以根据状态转移方程来计算 dp[node]。

具体地,对于每个节点 node,我们有两种选择:

  1. 如果我们盗取了当前节点 node,则不能盗取其孩子节点。因此,最大金额为 node.val + dp[node.left.left] + dp[node.left.right] + dp[node.right.left] + dp[node.right.right]。
  2. 如果我们不盗取当前节点 node,则可以选择盗取其左孩子和右孩子。因此,最大金额为 dp[node.left] + dp[node.right]。

最后,我们返回根节点的最大金额,即 dp[root]。

这样,我们可以通过动态规划来高效地解决这个问题,确保小偷在不触发警报的情况下,一晚能够盗取的最高金额。

# Definition for a binary tree node.
class TreeNode:def __init__(self, val=0, left=None, right=None):self.val = valself.left = leftself.right = rightclass Solution:def rob(self, root: TreeNode) -> int:# 调用辅助函数并返回根节点的最大金额return max(self.dfs(root))# 定义一个辅助函数来计算盗取以root为根节点的子树所能获得的最大金额def dfs(self, node):if not node:return 0, 0  # 返回不盗取当前节点和盗取当前节点的最大金额# 递归计算左右子树的最大金额left_no_rob, left_rob = self.dfs(node.left)right_no_rob, right_rob = self.dfs(node.right)# 不盗取当前节点时,可以选择盗取左右子树的根节点或者不盗取左右子树的根节点no_rob = max(left_no_rob, left_rob) + max(right_no_rob, right_rob)# 盗取当前节点时,不能盗取左右子树的根节点rob = node.val + left_no_rob + right_no_robreturn no_rob, robroot = TreeNode(3,2,3)
root.left = TreeNode(2)
root.right = TreeNode(3)
root.left.right = TreeNode(3)
root.right.right = TreeNode(1)obj = Solution()print(obj.rob(root))

感想:这题有点变化,用到了二叉树,这里要先算子节点的最大值,再算父节点,所以这题要采用后序遍历。

这篇关于算法打卡day41的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/962731

相关文章

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

大林 PID 算法

Dahlin PID算法是一种用于控制和调节系统的比例积分延迟算法。以下是一个简单的C语言实现示例: #include <stdio.h>// DALIN PID 结构体定义typedef struct {float SetPoint; // 设定点float Proportion; // 比例float Integral; // 积分float Derivative; // 微分flo

LeetCode 算法:二叉树的中序遍历 c++

原题链接🔗:二叉树的中序遍历 难度:简单⭐️ 题目 给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。 示例 1: 输入:root = [1,null,2,3] 输出:[1,3,2] 示例 2: 输入:root = [] 输出:[] 示例 3: 输入:root = [1] 输出:[1] 提示: 树中节点数目在范围 [0, 100] 内 -100 <= Node.

【Java算法】滑动窗口 下

​ ​    🔥个人主页: 中草药 🔥专栏:【算法工作坊】算法实战揭秘 🦌一.水果成篮 题目链接:904.水果成篮 ​ 算法原理 算法原理是使用“滑动窗口”(Sliding Window)策略,结合哈希表(Map)来高效地统计窗口内不同水果的种类数量。以下是详细分析: 初始化:创建一个空的哈希表 map 用来存储每种水果的数量,初始化左右指针 left

ROS2从入门到精通4-4:局部控制插件开发案例(以PID算法为例)

目录 0 专栏介绍1 控制插件编写模板1.1 构造控制插件类1.2 注册并导出插件1.3 编译与使用插件 2 基于PID的路径跟踪原理3 控制插件开发案例(PID算法)常见问题 0 专栏介绍 本专栏旨在通过对ROS2的系统学习,掌握ROS2底层基本分布式原理,并具有机器人建模和应用ROS2进行实际项目的开发和调试的工程能力。 🚀详情:《ROS2从入门到精通》 1 控制插

算法与数据结构面试宝典——回溯算法详解(C#,C++)

文章目录 1. 回溯算法的定义及应用场景2. 回溯算法的基本思想3. 递推关系式与回溯算法的建立4. 状态转移方法5. 边界条件与结束条件6. 算法的具体实现过程7. 回溯算法在C#,C++中的实际应用案例C#示例C++示例 8. 总结回溯算法的主要特点与应用价值 回溯算法是一种通过尝试各种可能的组合来找到所有解的算法。这种算法通常用于解决组合问题,如排列、组合、棋盘游

【图像识别系统】昆虫识别Python+卷积神经网络算法+人工智能+深度学习+机器学习+TensorFlow+ResNet50

一、介绍 昆虫识别系统,使用Python作为主要开发语言。通过TensorFlow搭建ResNet50卷积神经网络算法(CNN)模型。通过对10种常见的昆虫图片数据集(‘蜜蜂’, ‘甲虫’, ‘蝴蝶’, ‘蝉’, ‘蜻蜓’, ‘蚱蜢’, ‘蛾’, ‘蝎子’, ‘蜗牛’, ‘蜘蛛’)进行训练,得到一个识别精度较高的H5格式模型文件,然后使用Django搭建Web网页端可视化操作界面,实现用户上传一

【数据结构与算法 经典例题】使用队列实现栈(图文详解)

💓 博客主页:倔强的石头的CSDN主页               📝Gitee主页:倔强的石头的gitee主页    ⏩ 文章专栏:《数据结构与算法 经典例题》C语言                                   期待您的关注 ​​ 目录  一、问题描述 二、前置知识 三、解题思路 四、C语言实现代码 🍃队列实现代码:

算法11—判断一个树是不是二叉查询树

问题: 给定一个二叉树,判断它是否是二叉查询树。 思路: 要判断是否是二叉查询树,标准就是看每一个节点是否满足:1、左节点及以下节点的值比它小;2、右节点及以下节点的值比它大。当然,前提是子节点都存在的情况。所以,我们需要从根节点不断向下递归,只要所有节点都满足,那么就是BST,否则,就不是。 代码: [java]  view plain copy pri