北大硕士LeetCode算法专题课–递归和回溯

 算法专题课





北大硕士LeetCode算法专题课-栈、队列相关问题_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课–链表相关问题_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课-查找相关问题_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课-字符串相关问题_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课-数组相关问题_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课-基础算法查找_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课-基础算法之排序_骨灰级收藏家的博客-CSDN博客

北大硕士LeetCode算法专题课—算法复杂度介绍_骨灰级收藏家的博客-CSDN博客

二叉树简

二叉树(Binary Tree) 是一种树形数据结构,其中每个父节点最多可以有两个子节点。 二叉树的每个节点(node)包含三个属性:

data 数据

left 左子节点的地址

right 右子节点的地址

满二叉树(Full Binary Tree)每个结点要么没有子结点,要么有两个子结点

 

完美二叉树(Pefect Binary Tree)每个结点都有两个子结点,所有叶子结点都在同一层

完全二叉树(Complete Binary Tree)从根结点到倒数第二层为完美二叉树,最后一层可以不完全填充,其叶子结点都靠左对齐

 

二叉树天然的具有递归结构,二叉树的递归定义为:二叉树是一棵空树,或者是一棵由一个根节点和两棵互不相交的,  分别称作根的左子树和右子树组成的非空树;左子树和右子树又同样都是二叉树

 

 

二叉树的遍历方

LeetCode 题目中,二叉树的遍历方式是最基本,也是最重要的一类题目。先介绍一下二叉树的遍历方式。

先序遍历(前序遍历):按照根节点 -> 左孩子 -> 右孩子 的方式遍历,即「先序遍历」,每次先遍历根节点,  遍历结果为 1 2 4 5 3 6 7

 

LeetCode 题目中,二叉树的遍历方式是最基本,也是最重要的一类题目。先介绍一下二叉树的遍历方式。 中序遍历:按照左孩子 -> 根节点 -> 右孩子 的方式遍历,即「中序序遍历」

遍历结果为 4 2 5 1 6 3 7

LeetCode 题目中,二叉树的遍历方式是最基本,也是最重要的一类题目。先介绍一下二叉树的遍历方式。 后序遍历:按照左孩子 -> 右孩子 -> 根节点 的方式遍历,即「后序序遍历」

遍历结果为 4 5 2 6 7 3 1

LeetCode 144.二叉树的前序遍历 输出结果 [1,2,4,5,3,6]

 

def preorderTraversal2(root):  res = list()
if not root:
return res

stack = []  node = root
while stack or node:
while node:
res.append(node.val)  stack.append(node)  node = node.left
node = stack.pop()  node = node.right
return res

二叉树的遍历解题方法- 迭代使用

LeetCode 144.二叉树的前序遍历 输出结果 [1,2,4,5,3,6,7]

def preorderTraversal2(root):  res = list()
if not root:
return res

stack = []  node = root
while stack or node:
while node:
res.append(node.val)  stack.append(node)  node = node.left
node = stack.pop()  node = node.right
return res

二叉树的遍历解题方

LeetCode 102 中等.二叉树的层序遍历,使用队列

输出结果 [[1], [2, 3], [4, 5, 6, 7]]

 

二叉树的最大深度 (LeetCode 104)

给定一个二叉树,找出其最大深度。

二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。 说明: 叶子节点是指没有子节点的节点。

如果我们知道了左子树和右子树的最大深度 l r,那么该二叉树的最大深度即为 max(l,r)+1

 

如果我们知道了左子树和右子树的最大深度 l r,那么该二叉树的最大深度即为 max(l,r)+1

def maxDepth(root):  if root is None:
return 0
else:
left_height = maxDepth(root.left)  right_height = maxDepth(root.right)  return max(left_height, right_height) + 1

复杂度分析

时间复杂度O(n) ,其中 n 为二叉树节点的个数。每个节点在递归中只被遍历一次。 空间复杂度O(height) ,其中 height表示二叉树的高度。递归函数需要栈空间,而栈空 间取决于递归的深度,因此空间复杂度等价于二叉树的高度。

翻转二叉 (LeetCode 226)

给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。

给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。

 给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。 

def invertTree(root):
# 递归终止条件,节点为空时返回
if not root:
return None
# 递归调用
left = invertTree(root.left)  right = invertTree(root.right)
# 将当前节点的左右子树交换 root.left, root.right = right, left  return root

复杂度分析:时间复杂度:O(n) 空间复杂度:O(n)

路径总 (LeetCode 112)

给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径,  这条路径上所有节点值相加等于目标和 targetSum 。如果存在,返回 true ;否则,返回 false

叶子节点 是指没有子节点的节点。

输入root = [5,4,8,11,null,13,4,7,2,null,null,null,1], targetSum = 22

输出true

解释:等于目标和的根节点到叶节点路径如上图所示。

使用递归的思路解决问题的时候, 需要注意递归的终止条件。

class Solution(object):
def hasPathSum(self, root, targetSum):  """
:type root: TreeNode
:type targetSum: int
:rtype: bool  """
if not root:
return targetSum == 0
if self.hasPathSum(root.left, targetSum - root.val):
return True
if self.hasPathSum(root.right, targetSum - root.val):
return True
return False

 使用递归的思路解决问题的时候, 需要注意递归的终止条件。下面写法中的递归终止条件考虑并不全面,  当输入为[], targetSum = 0 时, 下面的解法会返回True 但实际上应该是False

class Solution(object):
def hasPathSum(self, root, targetSum):  """
:type root: TreeNode
:type targetSum: int
:rtype: bool  """
if not root:
return targetSum == 0
if self.hasPathSum(root.left, targetSum - root.val):
return True
if self.hasPathSum(root.right, targetSum - root.val):
return True
return False

电话号码的字母组 (LeetCode 17)

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

给给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

 

digits 是数字字符串, s(digits) digits 所能代表的字母字符串

s(digits[0…n-1]) = letter(digits[0]) + s(digits[1…n-1]) = letter(digits[0]) + letter(digits[1]) + s(digits[2…n-1])

复杂度分析

时间复杂度O(3n) (O(2n))

空间复杂度O(m+n)

回溯

回溯法也称试探法,它的基本思想是:

从问题的某一种状态(初始状态)出发,搜索从这种状态出发所能达到的所有“状态”,

当一条路走到“尽头”的时候(不能再前进),再后退一步或若干步,从另一种可能“状态”出发,继续搜索 直到所有的“路径”(状态)都试探过

这种不断“前进”、不断“回溯”寻找解的方法,就称作“回溯法”

回溯法是暴力解法的一个主要手段

全排 (LeetCode 46)

给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。

 

 

def permute(nums):
def backtrack(first=0):  if first == n:
res.append(nums[:])
for i in range(first, n):
nums[first], nums[i] = nums[i], nums[first]  backtrack(first + 1)
nums[first], nums[i] = nums[i], nums[first]  n = len(nums)
res = []  backtrack()  return res

 

文章出处登录后可见!

已经登录?立即刷新

共计人评分,平均

到目前为止还没有投票!成为第一位评论此文章。

(0)
社会演员多的头像社会演员多普通用户
上一篇 2023年12月28日
下一篇 2023年12月28日

相关推荐