Merge pull request #2807 from gazeldx/binary_tree_iteration

二叉树的迭代遍历.md 加入 Python 版本的后序遍历新解法
This commit is contained in:
程序员Carl
2024-12-05 10:04:21 +08:00
committed by GitHub

View File

@ -240,14 +240,14 @@ class Solution {
# 前序遍历-迭代-LC144_二叉树的前序遍历
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:
@ -262,19 +262,23 @@ class Solution:
# 中序遍历-迭代-LC94_二叉树的中序遍历
class Solution:
def inorderTraversal(self, root: TreeNode) -> List[int]:
stack = [] # 不能提前将root结点加入stack中
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
```
@ -289,7 +293,7 @@ class Solution:
result = []
while stack:
node = stack.pop()
# 中点先处理
# 中点先处理
result.append(node.val)
# 左孩子先入栈
if node.left:
@ -301,6 +305,44 @@ class Solution:
return result[::-1]
```
#### Python 后序遍历的迭代新解法:
* 本解法不同于前文介绍的`逆转前序遍历调整后的结果`而是采用了对每个节点直接处理这个实现方法在面试中不容易写出来在下一节我将改造本代码奉上代码更简洁更套路化更容易实现的统一方法
```python
class Solution:
def postorderTraversal(self, root: Optional[TreeNode]) -> List[int]:
values = []
stack = []
popped_nodes = set() # 记录值已经被收割了的 nodes这是关键已经被收割的节点还在树中还会被访问到但逻辑上已经等同于 null 节点。
current = root
while current or stack:
if current: # 一次处理完一个节点和他的左右儿子节点,不处理孙子节点,孙子节点由左右儿子等会分别处理。
stack.append(current) # 入栈自己
if current.right:
stack.append(current.right) # 入栈右儿子
if current.left: # 因为栈是后进先出,后序是‘左右中’,所以后加左儿子
stack.append(current.left) # 入栈左儿子
current = None # 会导致后面A处出栈
continue
node = stack.pop() # A处出的是左儿子如果无左儿子出的就是右儿子如果连右儿子也没有出的就是自己了。
# 如果 node 是叶子节点,就可以收割了;如果左右儿子都已经被收割了,也可以收割
if (node.left is None or node.left in popped_nodes) and \
(node.right is None or node.right in popped_nodes):
popped_nodes.add(node)
values.append(node.val)
continue
current = node # 不符合收割条件,说明 node 下还有未入栈的儿子,就去入栈
return values
```
### Go
> 迭代法前序遍历