二叉树实战篇1

前言

上文带大家学习了二叉树的理论基础,如果没看过的点这去回顾下 ,今天带大家进行二叉树的实战篇1,学会如何去遍历二叉树,无论什么要求怎么遍历,一文带大家弄懂。本文用于记录自己的学习过程,同时向大家进行分享相关的内容。本文内容参考于代码随想录同时包含了自己的许多学习思考过程,如果有错误的地方欢迎批评指正!

二叉树实战篇1

二叉树的递归遍历

说到二叉树的递归遍历,这里正好带大家巩固下遍历的知识。很多人可能对于到递归就是一听就会,一写就废的。那么为什么这样呢?因为递归就是将复杂问题化为简单的重复的小问题,一听上去好像我们会了,真的写起来却发现自己写不出来,这就是递归三要素没有彻底的搞清楚。什么是递归三要素:

  • 确定递归的参数和返回值:我们我们递归需要传递进什么参数去继续处理,并且还要确定函数的返回值是什么,进而确定函数的返回类型。
  • 确定递归的终止条件:这是很重要的,我们什么时候递归终止可以不用在递归了,否则就将会陷入递归无限的死循环中。这会导致系统的栈溢出情况。
  • 确定单层递归的逻辑:确定每一层递归我们需要处理什么内容,同时在这也会重复调用自己来实现递归过程。

那么我们就以二叉树的递归遍历来联手

前序遍历

再次回顾下,前序遍历的顺序是根左右,先根节点,在左右节点。所以就很直观了,我们先保存根节点值,在进入左子树保存,直至节点为空结束递归。后面的中序遍历和后序遍历是同样的道理,就不再叙述了,直接放代码。

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right  class Solution:     def preorderTraversal(self, root: TreeNode) -> List[int]:         res = []                  def dfs(node):             if node is None:                 return                          res.append(node.val)             dfs(node.left)             dfs(node.right)         dfs(root)         return res 

中序遍历

class Solution:     def inorderTraversal(self, root: TreeNode) -> List[int]:         res = []                  def dfs(node):             if node is None:                 return                          dfs(node.left)             res.append(node.val)             dfs(node.right)         dfs(root)         return res 

后续遍历

class Solution:     def postorderTraversal(self, root: TreeNode) -> List[int]:         res = []                  def dfs(node):             if node is None:                 return                          dfs(node.left)             dfs(node.right)             res.append(node.val)          dfs(root)         return res 

二叉树的迭代遍历

二叉树的递归遍历代码很简单,也很容易去理解的,那么我们再来看看,若是不用递归的方式去前中后序遍历该怎么去进行,这就是这节要说的二叉树的迭代遍历。

前序遍历

我们来看前序遍历的顺序是根左右,那么我们通过栈来进行操作,先根节点进栈,然后根节点出栈,然后重点来了,先进右节点在进左节点。为什么要这样?因为我们顺序是根左右,栈的特性后进先出,那么就要右节点先进栈,左节点在进栈,然后在以进栈的左节点作为根节点重复上述操作,直到将栈清空为止。实现代码为:

class Solution:     def preorderTraversal(self, root: TreeNode) -> List[int]:         # 根节点为空则返回空列表         if not root:             return []         stack = [root]         result = []         while stack:             node = stack.pop()             # 中节点先处理             result.append(node.val)             # 右孩子先入栈             if node.right:                 stack.append(node.right)             # 左孩子后入栈             if node.left:                 stack.append(node.left)         return result 

中序遍历

刚才讲了前序遍历的迭代法,那么我们是否可以以相同的方式来进行呢?当然不行?我们来看为什么,是因为前序遍历的顺序是中左右,先访问的元素是中间节点,要处理的元素也是中间节点,所以刚刚才能写出相对简洁的代码,因为要访问的元素和要处理的元素顺序是一致的,都是中间节点。

那么中序遍历就是先从根节点开始进栈,往左节点一直进栈,直至为空,然后弹出其栈顶的节点返回,再看其有没有右节点有就继续进栈,在左节点遍历直至空,重复操作即可。

class Solution:     def inorderTraversal(self, root: TreeNode) -> List[int]:          if not root:             return []         stack = []  # 不能提前将root节点加入stack中          result = []         cur = root         while cur or stack:             # 先迭代访问最底层的左子树节点             if cur:                      stack.append(cur)                 cur = cur.left		             # 到达最左节点后处理栈顶节点                 else:		                 cur = stack.pop()                 result.append(cur.val)                 # 取栈顶元素右节点                 cur = cur.right	         return result 

后序遍历

后序遍历就比较简单了,我们可以参考前序遍历,前序遍历的顺序是根左右,后序遍历的顺序是左右根,那我们可以按照前序遍历的方法,不过换一下先进左节点在进右节点,就可以得到根右左的顺序,我们再将他倒序即可得到左右根。

class Solution:     def postorderTraversal(self, root: TreeNode) -> List[int]:         if not root:             return []         stack = [root]         result = []         while stack:             node = stack.pop()             # 中节点先处理             result.append(node.val)             # 左孩子先入栈             if node.left:                 stack.append(node.left)             # 右孩子后入栈             if node.right:                 stack.append(node.right)         # 将最终的数组翻转         return result[::-1] 

二叉树的层序遍历

层序遍历(学会这个,以一打十)

102. 二叉树的层序遍历 - 力扣(LeetCode)

二叉树实战篇1

相关技巧:层序遍历就是一层的所有节点进行输出,我们可以借助队列来实现,根节点进入队列,其子节点进入队列,然后根节点出队列记录。我们这里可以借助一个变量level来记录其所属的不同层。所以其实现代码如下:

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def levelOrder(self, root: Optional[TreeNode]) -> List[List[int]]:         if not root:             return []         queue = collections.deque([root])         result = []         while queue:             level = []             for _ in range(len(queue)):                 cur = queue.popleft()                 level.append(cur.val)                 if cur.left:                     queue.append(cur.left)                 if cur.right:                     queue.append(cur.right)             result.append(level)         return result 

层序遍历(倒序)

107. 二叉树的层序遍历 II - 力扣(LeetCode)

直接按照层序遍历的结果倒序即可,非常简单。

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def levelOrderBottom(self, root: TreeNode) -> List[List[int]]:         if not root:             return []         queue = collections.deque([root])         result = []         while queue:             level = []             for _ in range(len(queue)):                 cur = queue.popleft()                 level.append(cur.val)                 if cur.left:                     queue.append(cur.left)                 if cur.right:                     queue.append(cur.right)             result.append(level)         return result[::-1] 

二叉树的右视图

199. 二叉树的右视图 - 力扣(LeetCode)

层序遍历,每次输出每层的最后一个节点就行了。

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def rightSideView(self, root: Optional[TreeNode]) -> List[int]:         if not root:             return []                  queue=collections.deque([root])         right_view=[]         while queue:             levels=len(queue)             for i in range(levels):                 node=queue.popleft()                  if i == levels-1:                     right_view.append(node.val)                  if node.left:                     queue.append(node.left)                 if node.right:                     queue.append(node.right)         return right_view 

二叉树的层平均值

637. 二叉树的层平均值 - 力扣(LeetCode)

每层求和求平均即可

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def averageOfLevels(self, root: TreeNode) -> List[float]:         if not root:             return []          queue = collections.deque([root])         averages = []                  while queue:             size = len(queue)             level_sum = 0                          for i in range(size):                 node = queue.popleft()                                                   level_sum += node.val                                      if node.left:                     queue.append(node.left)                 if node.right:                     queue.append(node.right)                          averages.append(level_sum / size)                  return averages 

N叉树的层序遍历

429. N 叉树的层序遍历 - 力扣(LeetCode)

二叉树实战篇1

""" # Definition for a Node. class Node:     def __init__(self, val: Optional[int] = None, children: Optional[List['Node']] = None):         self.val = val         self.children = children """  class Solution:     def levelOrder(self, root: 'Node') -> List[List[int]]:         if not root:             return []          result = []         queue = collections.deque([root])          while queue:             level_size = len(queue)             level = []              for _ in range(level_size):                 node = queue.popleft()                 level.append(node.val)                  for child in node.children:                     queue.append(child)              result.append(level)          return result 

在每行中找出最大值

515. 在每个树行中找最大值 - 力扣(LeetCode)

在每层找最大值,作比较记录即可

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def largestValues(self, root: TreeNode) -> List[int]:         if not root:             return []          result = []         queue = collections.deque([root])          while queue:             level_size = len(queue)             max_val = float('-inf')              for _ in range(level_size):                 node = queue.popleft()                 max_val = max(max_val, node.val)                  if node.left:                     queue.append(node.left)                  if node.right:                     queue.append(node.right)              result.append(max_val)          return result 

填充每个节点的下一个右侧节点指针

通过层序遍历,然后每层内指向下个节点即可,这里需要个额外的指针记录上个节点

二叉树实战篇1

""" # Definition for a Node. class Node:     def __init__(self, val: int = 0, left: 'Node' = None, right: 'Node' = None, next: 'Node' = None):         self.val = val         self.left = left         self.right = right         self.next = next """  class Solution:     def connect(self, root: 'Node') -> 'Node':         if not root:             return root                  queue = collections.deque([root])                  while queue:             level_size = len(queue)             prev = None                          for i in range(level_size):                 node = queue.popleft()                                  if prev:                     prev.next = node                                  prev = node                                  if node.left:                     queue.append(node.left)                                  if node.right:                     queue.append(node.right)                  return root 

填充每个节点的下一个右侧节点指针 II

117. 填充每个节点的下一个右侧节点指针 II - 力扣(LeetCode)

这里的不同就是上个满二叉树,这里并不是满二叉树,不过原理逻辑相同。

二叉树实战篇1

""" # Definition for a Node. class Node:     def __init__(self, val: int = 0, left: 'Node' = None, right: 'Node' = None, next: 'Node' = None):         self.val = val         self.left = left         self.right = right         self.next = next """  class Solution:     def connect(self, root: 'Node') -> 'Node':         if not root:             return root                  queue = collections.deque([root])                  while queue:             level_size = len(queue)             prev = None                          for i in range(level_size):                 node = queue.popleft()                                  if prev:                     prev.next = node                                  prev = node                                  if node.left:                     queue.append(node.left)                                  if node.right:                     queue.append(node.right)                  return root 

二叉树的最大深度

104. 二叉树的最大深度 - 力扣(LeetCode)

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def maxDepth(self, root: TreeNode) -> int:         if not root:             return 0                  depth = 0         queue = collections.deque([root])                  while queue:             depth += 1             for _ in range(len(queue)):                 node = queue.popleft()                 if node.left:                     queue.append(node.left)                 if node.right:                     queue.append(node.right)                  return depth 

二叉树的最小深度

111. 二叉树的最小深度 - 力扣(LeetCode)

二叉树实战篇1

# Definition for a binary tree node. # class TreeNode: #     def __init__(self, val=0, left=None, right=None): #         self.val = val #         self.left = left #         self.right = right class Solution:     def minDepth(self, root: TreeNode) -> int:         if not root:             return 0         depth = 0         queue = collections.deque([root])                  while queue:             depth += 1              for _ in range(len(queue)):                 node = queue.popleft()                                  if not node.left and not node.right:                     return depth                              if node.left:                     queue.append(node.left)                                      if node.right:                     queue.append(node.right)          return depth 

算法基础系列

一文了解什么是数组及其经典考察题目
走进链表及其经典考察题目
还不知道什么是哈希表,看这篇文章就够了
字符串匹配究极大招【KMP】:带你一步步从原理到构建
【栈与队列】:基础实战篇
【双指针法】:这么常用的你怎么能不知道 - carpell - 博客园
【二叉树】理论基础篇1 - carpell - 博客园

发表评论

您必须 [ 登录 ] 才能发表留言!

相关文章