目录
- 前言
- 二叉树的递归遍历
- 二叉树的迭代遍历
- 二叉树的层序遍历
- 层序遍历(学会这个,以一打十)
- 层序遍历(倒序)
- 二叉树的右视图
- 二叉树的层平均值
- N叉树的层序遍历
- 在每行中找出最大值
- 填充每个节点的下一个右侧节点指针
- 填充每个节点的下一个右侧节点指针 II
- 二叉树的最大深度
- 二叉树的最小深度
- 算法基础系列
前言
上文带大家学习了二叉树的理论基础,如果没看过的点这去回顾下 ,今天带大家进行二叉树的实战篇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)
相关技巧:层序遍历就是一层的所有节点进行输出,我们可以借助队列来实现,根节点进入队列,其子节点进入队列,然后根节点出队列记录。我们这里可以借助一个变量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)
直接按照层序遍历的结果倒序即可,非常简单。
- # 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)
层序遍历,每次输出每层的最后一个节点就行了。
- # 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)
每层求和求平均即可
- # 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)
- """
- # 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)
在每层找最大值,作比较记录即可
- # 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
复制代码 填充每个节点的下一个右侧节点指针
通过层序遍历,然后每层内指向下个节点即可,这里需要个额外的指针记录上个节点
- """
- # 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)
这里的不同就是上个满二叉树,这里并不是满二叉树,不过原理逻辑相同。
- """
- # 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)
- # 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)
- # 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 - 博客园
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |