苏州市建设交通高等学校网站,韩雪冬做网站多少钱,响应式网站做seo,如何自建网页112 路径总和
给定一个二叉树和一个目标和#xff0c;判断该树中是否存在根节点到叶子节点的路径#xff0c;这条路径上所有节点值相加等于目标和。 说明: 叶子节点是指没有子节点的节点。 示例: 给定如下二叉树#xff0c;以及目标和 sum 22#xff0c; 返回 true, 因为…112 路径总和
给定一个二叉树和一个目标和判断该树中是否存在根节点到叶子节点的路径这条路径上所有节点值相加等于目标和。 说明: 叶子节点是指没有子节点的节点。 示例: 给定如下二叉树以及目标和 sum 22 返回 true, 因为存在目标和为 22 的根节点到叶子节点的路径 5-4-11-2。
思路
参考https://www.programmercarl.com/0112.%E8%B7%AF%E5%BE%84%E6%80%BB%E5%92%8C.html#%E6%80%9D%E8%B7%AF 递归函数什么时候要有返回值什么时候没有返回值特别是有的时候递归函数返回类型为bool类型。 那么接下来我通过详细讲解如下两道题来回答这个问题 112.路径总和 113.路径总和ii 这道题我们要遍历从根节点到叶子节点的路径看看总和是不是目标和。
递归
可以使用深度优先遍历的方式本题前中后序都可以无所谓因为中节点也没有处理逻辑来遍历二叉树
确定递归函数的参数和返回类型
参数需要二叉树的节点还需要一个计数器这个计数器用来计算二叉树的一条边之和是否正好是目标和计数器为int型。 再来看返回值递归函数什么时候需要返回值什么时候不需要返回值这里总结如下三点
如果需要搜索整棵二叉树且不用处理递归返回值递归函数就不要返回值。这种情况就是本文下半部分介绍的113.路径总和ii如果需要搜索整棵二叉树且需要处理递归返回值递归函数就需要返回值。 这种情况我们在236. 二叉树的最近公共祖先 中介绍如果要搜索其中一条符合条件的路径那么递归一定需要返回值因为遇到符合条件的路径了就要及时返回。本题的情况
而本题我们要找一条符合条件的路径所以递归函数需要返回值及时返回那么返回类型是什么呢 如图所示 图中可以看出遍历的路线并不要遍历整棵树只需要有符合题意得时候 立刻返回就行所以递归函数需要返回值可以用bool类型表示。 所以代码如下
# 我的问题就是遍历到一条路径底部时候 如何转到另一条路径 —— 回溯def travelsal(self, node, count): # 递归第一步 参数为当前节点 另外采用一个计数器 遍历路径时候减去当前遍历节点的值在python中 函数头不明确指明返回值类型 但自己要知道 返回值什么 为什么需要返回值
确定终止条件
首先计数器如何统计这一条路径的和呢 不要去累加然后判断是否等于目标和那么代码比较麻烦可以用递减让计数器count初始为目标和然后每次减去遍历路径节点上的数值。逆向思维 如果最后count 0同时到了叶子节点的话说明找到了目标和。 如果遍历到了叶子节点count不为0就是没找到。 递归终止条件代码如下
if not node.left and not node.right and count 0:return True # 到达叶子节点 且count为0 说明当前路径刚好符合目标
if not node.left and not node.right:return False # 到达叶子节点 但是count不为0 因为路径有很多条 到达路径时候必然是叶子节点确定单层递归的逻辑
因为终止条件是判断叶子节点所以递归的过程中就不要让空节点进入递归了。 递归函数是有返回值的如果递归函数返回true说明找到了合适的路径应该立刻返回。 代码如下
if node.left:count - node.left.valif self.travelsal(node.left, count): # 递归处理节点return Truecount node.left.val # 回溯
# 右
if node.right:count - node.right.valif self.travelsal(node.right, count):return Truecount node.right.val # 回溯完整代码
class TreeNode(object):def __init__(self, val0, leftNone, rightNone):self.val valself.left leftself.right rightclass Solution(object):def hasPathSum(self, root, targetSum)::type root: TreeNode:type targetSum: int:rtype: boolif not root:return False# self.travelsal(root, targetSum)return self.travelsal(root, targetSum - root.val) # 减去根节点的 再遍历之后的# 我的问题就是遍历到一条路径底部时候 如何转到另一条路径 —— 回溯def travelsal(self, node, count): # 递归第一步 参数为当前节点 另外采用一个计数器 遍历路径时候减去当前遍历节点的值if not node.left and not node.right and count 0:return True # 到达叶子节点 且count为0 说明当前路径刚好符合目标if not node.left and not node.right:return False # 到达叶子节点 但是count不为0 因为路径有很多条 到达路径时候必然是叶子节点# 左# if node.left:# count - node.left.val# self.travelsal(node.left, count)# count node.left.val # 回溯 # 上面错误if node.left:count - node.left.valif self.travelsal(node.left, count): # 递归处理节点return Truecount node.left.val # 回溯 # 右if node.right:count - node.right.valif self.travelsal(node.right, count):return Truecount node.right.val # 回溯return False # 注意最后还要返回False 因为没有符合以上True的情况迭代法
如果使用栈模拟递归的话那么如果做回溯呢 此时栈里一个元素不仅要记录该节点指针还要记录从头结点到该节点的路径数值总和。
# 法二 迭代法
class Solution(object):def hasPathSum(self, root, targetSum):if not root:return False # 树为空 直接返回False# 采用前序遍历 中左右 stack [(root, root.val)] # 栈里面存储的是节点及节点值while stack:node, path_sum stack.pop()# 如果该节点是叶子节点了同时该节点的路径数值等于sum那么就返回trueif not node.left and not node.right and path_sum targetSum:return True# # 左节点压进去一个节点的时候将该节点的路径数值也记录下来if node.left:stack.append((node.left, path_sum node.left.val))# 右节点if node.right:stack.append((node.right, path_sum node.right.val))return False113 路径总和2
给定一个二叉树和一个目标和找到所有从根节点到叶子节点路径总和等于给定目标和的路径。本题与上一题得区别是要找到所有的路径 说明: 叶子节点是指没有子节点的节点。 示例: 给定如下二叉树以及目标和 sum 22
思路
113.路径总和ii要遍历整个树找到所有路径所以递归函数不要返回值 如图 代码
class TreeNode(object):def __init__(self, val0, leftNone, rightNone):self.val valself.left leftself.right right# 本题与112题不同的是 本题要找出所有路径 所以除了count计数之外 还需要一个列表来存储路径结果
class Solution(object):def __init__(self):self.result [] # 总的结果self.path [] # 单条路径结果def pathSum(self, root, targetSum)::type root: TreeNode:type targetSum: int:rtype: List[List[int]]# 先清空 result 和 path# self.result.clear()# self.path.clear()del self.result[:] # 注意若是 del self.result则是删除整个对象 而加了[:]是删除其中的元素del self.path[:]if not root:return self.resultself.path.append(root.val) # 把根节点放入pathself.travelsal(root, targetSum - root.val)return self.resultdef travelsal(self, node, count): # 用node表示更一般化 容易理解if not node.left and not node.right and count 0:self.result.append(self.path[:]) # 收获结果 符合题目的单条路径加到总结果中 注意是 self.path[:]return if not node.left and not node.right:returnif node.left: # 左节点不为空 空节点不遍历self.path.append(node.left.val)count - node.left.valself.travelsal(node.left, count) # 递归节点count node.left.val # 回溯self.path.pop() # 回溯 pop不需要传入参数if node.right: # 右节点不为空 空节点不遍历self.path.append(node.right.val)count - node.right.valself.travelsal(node.right, count) # 递归节点count node.right.val # 回溯self.path.pop() # 回溯return # 不需要返回值 直接一个return就行self.result.append(self.path) 与 self.result.append(self.path[:]) 的区别
result.append(self.path) 将 self.path 添加到 result 列表中但实际上它是对 self.path 的引用。这意味着如果后续更改了 self.path 的值result 列表中的相应元素也会受到影响因为它们引用相同的对象。这是因为 self.path 是一个可变对象例如列表、字典并且在列表中仅存储了对该对象的引用。
self.path [1, 2, 3]
result []
result.append(self.path)self.path.append(4)
print(result) # 输出 [1, 2, 3, 4]
result.append(self.path[:]) 创造了 self.path 的一个副本并将该副本添加到 result 列表中。这意味着即使后续更改了 self.path 的值result 列表中的元素仍然保持不变因为它们引用的是一个独立的副本。
self.path [1, 2, 3]
result []
result.append(self.path[:])self.path.append(4)
print(result) # 输出 [1, 2, 3]
总结
本篇通过leetcode上112. 路径总和 和 113. 路径总和ii 详细的讲解了 递归函数什么时候需要返回值什么不需要返回值。 这两道题目是掌握这一知识点非常好的题目大家看完本篇文章再去做题就会感受到搜索整棵树和搜索某一路径的差别。 对于112. 路径总和我依然给出了递归法和迭代法这种题目其实用迭代法会复杂一些能掌握递归方式就够了
参考https://www.programmercarl.com/0112.%E8%B7%AF%E5%BE%84%E6%80%BB%E5%92%8C.html