LeetCodeDiary

A Diary for solving LeetCode problems

View on GitHub

987. 二叉树的垂序遍历

给你二叉树的根结点 root ,请你设计算法计算二叉树的 垂序遍历 序列。

对位于 (row, col) 的每个结点而言,其左右子结点分别位于 (row + 1, col - 1)(row + 1, col + 1) 。树的根结点位于 (0, 0)

二叉树的 垂序遍历 从最左边的列开始直到最右边的列结束,按列索引每一列上的所有结点,形成一个按出现位置从上到下排序的有序列表。如果同行同列上有多个结点,则按结点的值从小到大进行排序。

返回二叉树的 垂序遍历 序列。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[9],[3,15],[20],[7]]
解释:
列 -1 :只有结点 9 在此列中。
列  0 :只有结点 3 和 15 在此列中,按从上到下顺序。
列  1 :只有结点 20 在此列中。
列  2 :只有结点 7 在此列中。

示例 2:

img

输入:root = [1,2,3,4,5,6,7]
输出:[[4],[2],[1,5,6],[3],[7]]
解释:
列 -2 :只有结点 4 在此列中。
列 -1 :只有结点 2 在此列中。
列  0 :结点 1 、5 和 6 都在此列中。
          1 在上面,所以它出现在前面。
          5 和 6 位置都是 (2, 0) ,所以按值从小到大排序,5 在 6 的前面。
列  1 :只有结点 3 在此列中。
列  2 :只有结点 7 在此列中。

示例 3:

img

输入:root = [1,2,3,4,6,5,7]
输出:[[4],[2],[1,5,6],[3],[7]]
解释:
这个示例实际上与示例 2 完全相同,只是结点 5 和 6 在树中的位置发生了交换。
因为 5 和 6 的位置仍然相同,所以答案保持不变,仍然按值从小到大排序。

提示:

树 困难

层序遍历输出即可

代码

# 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 verticalTraversal(self, root: TreeNode) -> List[List[int]]:
        queue = deque([(root, 0)])
        d = defaultdict(list)
        while queue:
            n = len(queue)
            td = defaultdict(list) # 该层临时字典
            for i in range(n):
                cur, col = queue.popleft()
                # d[col].append(cur.val)
                insort(td[col], cur.val)
                if cur.left:
                    queue.append((cur.left, col-1))
                if cur.right:
                    queue.append((cur.right, col+1))
            for key in td.keys():
                d[key].extend(td[key])
        kvs = list(d.items())
        kvs.sort(key=lambda k:k[0])
        return [key[1] for key in kvs]
# 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 verticalTraversal(self, root: TreeNode) -> List[List[int]]:
        if not root:
            return []
        
        def dfs(root, level, col):
            if not root:
                return
            nodelist.append((root.val, level, col))
            dfs(root.left, level+1, col-1)
            dfs(root.right, level+1, col+1)
        nodelist = []

        dfs(root, 0, 0)
        nodelist.sort(key=lambda k:(k[2],k[1],k[0]))
        print(nodelist)
        ans = []
        t = [nodelist[0][0]]
        for i in range(1,len(nodelist)):
            if nodelist[i][2] != nodelist[i-1][2]:
                ans.append(t[:])
                t = [nodelist[i][0]]
            else:
                t.append(nodelist[i][0])
        ans.append(t[:])
        return ans