617.合并二叉树
题目描述
给你两棵二叉树: root1 和 root2 。
想象一下,当你将其中一棵覆盖到另一棵之上时,两棵树上的一些节点将会重叠(而另一些不会)。你需要将这两棵树合并成一棵新二叉树。合并的规则是:如果两个节点重叠,那么将这两个节点的值相加作为合并后节点的新值;否则,不为 null 的节点将直接作为新二叉树的节点。
返回合并后的二叉树。
注意: 合并过程必须从两个树的根节点开始。
示例 1:
输入:root1 = [1,3,2,5], root2 = [2,1,3,null,4,null,7]
输出:[3,4,5,5,4,null,7]
思路
- 深度优先搜索
使用深度优先搜索合并两个二叉树。从根节点开始遍历两个二叉树,并将对应的节点进行合并。
两个二叉树的对应节点可能存在以下三种情况,对于每种情况使用不同的合并方式。
(1)如果两个二叉树的对应节点都为空,则合并后的二叉树的对应节点也为空;
(2)如果两个二叉树的对应节点只有一个为空,则合并后的二叉树的对应节点为其中的非空节点;
(3)如果两个二叉树的对应节点都不为空,则合并后的二叉树的对应节点的值为两个二叉树的对应节点的值之和,此时需要显性合并两个节点。
对一个节点进行合并之后,还要对该节点的左右子树分别进行合并。这是一个递归的过程。
2. 广度优先搜索
使用广度优先搜索合并两个二叉树。首先判断两个二叉树是否为空,如果两个二叉树都为空,则合并后的二叉树也为空,如果只有一个二叉树为空,则合并后的二叉树为另一个非空的二叉树。
如果两个二叉树都不为空,则首先计算合并后的根节点的值,然后从合并后的二叉树与两个原始二叉树的根节点开始广度优先搜索,从根节点开始同时遍历每个二叉树,并将对应的节点进行合并。
使用三个队列分别存储合并后的二叉树的节点以及两个原始二叉树的节点。初始时将每个二叉树的根节点分别加入相应的队列。每次从每个队列中取出一个节点,判断两个原始二叉树的节点的左右子节点是否为空。如果两个原始二叉树的当前节点中至少有一个节点的左子节点不为空,则合并后的二叉树的对应节点的左子节点也不为空。对于右子节点同理。
如果合并后的二叉树的左子节点不为空,则需要根据两个原始二叉树的左子节点计算合并后的二叉树的左子节点以及整个左子树。考虑以下两种情况:
(1)如果两个原始二叉树的左子节点都不为空,则合并后的二叉树的左子节点的值为两个原始二叉树的左子节点的值之和,在创建合并后的二叉树的左子节点之后,将每个二叉树中的左子节点都加入相应的队列;
(2)如果两个原始二叉树的左子节点有一个为空,即有一个原始二叉树的左子树为空,则合并后的二叉树的左子树即为另一个原始二叉树的左子树,此时也不需要对非空左子树继续遍历,因此不需要将左子节点加入队列。
对于右子节点和右子树,处理方法与左子节点和左子树相同。
代码
class TreeNode(object): # Definition for a binary tree node.
def __init__(self, val=0, left=None, right=None):
self.val = val
self.left = left
self.right = right
class Solution(object):
def mergeTrees(self, root1, root2):
"""
深度优先搜索
:type root1: TreeNode
:type root2: TreeNode
:rtype: TreeNode
"""
if not root1:
return root2
if not root2:
return root1
merged = TreeNode(root1.val + root2.val)
merged.left = self.mergeTrees(root1.left, root2.left)
merged.right = self.mergeTrees(root1.right, root2.right)
return merged
def mergeTrees1(self, root1, root2):
"""广度优先搜索"""
import collections
if not root1:
return root2
if not root2:
return root1
merged = TreeNode(root1.val + root2.val)
queue = collections.deque([merged])
queue1 = collections.deque([root1])
queue2 = collections.deque([root2])
while queue1 and queue2:
node = queue.popleft()
node1 = queue1.popleft()
node2 = queue2.popleft()
left1, right1 = node1.left, node1.right
left2, right2 = node2.left, node2.right
if left1 and left2:
left = TreeNode(left1.val + left2.val)
node.left = left
queue.append(left)
queue1.append(left1)
queue2.append(left2)
elif left1:
node.left = left1
elif left2:
node.left = left2
if right1 and right2:
right = TreeNode(right1.val + right2.val)
node.right = right
queue.append(right)
queue1.append(right1)
queue2.append(right2)
elif right1:
node.right = right1
elif right2:
node.right = right2
return merged
def preorder_traverse(self, root):
if not root:
return
print(root.val, end=" ")
self.preorder_traverse(root.left)
self.preorder_traverse(root.right)
if __name__ == "__main__":
slt = Solution()
root1 = TreeNode(1)
node12 = TreeNode(3)
node13 = TreeNode(2)
node14 = TreeNode(5)
root1.left = node12
root1.right = node13
node12.left = node14
print("root1:", end="[")
slt.preorder_traverse(root1)
print("]")
root2 = TreeNode(2)
node22 = TreeNode(1)
node23 = TreeNode(3)
node24 = TreeNode(4)
node25 = TreeNode(7)
root2.left = node22
root2.right = node23
node22.right = node24
node23.right = node25
print("root2:", end="[")
slt.preorder_traverse(root2)
print("]")
# merged_tree = slt.mergeTrees(root1, root2)
merged_tree = slt.mergeTrees1(root1, root2)
print("merger tree:", end="[")
slt.preorder_traverse(merged_tree)
print("]")