1. 程式人生 > 其它 >LeetCode——987. 二叉樹的垂序遍歷(Java)

LeetCode——987. 二叉樹的垂序遍歷(Java)

題目描述

題幹:
給你二叉樹的根結點 root ,請你設計演算法計算二叉樹的 垂序遍歷 序列。
對位於(row, col)的每個結點而言,
其左右子結點分別位於(row + 1, col - 1)和(row + 1, col + 1) 。
樹的根結點位於 (0, 0) 。
二叉樹的 垂序遍歷 從最左邊的列開始直到最右邊的列結束,
按列索引每一列上的所有結點,形成一個按出現位置從上到下排序的有序列表。
如果同行同列上有多個結點,則按結點的值從小到大進行排序。
返回二叉樹的 垂序遍歷 序列。

示例 1:
輸入:root = [3,9,20,null,null,15,7]
輸出:[[9],[3,15],[20],[7]]
解釋:
列 -1 :只有結點 9 在此列中。
列  0 :只有結點 3 和 15 在此列中,按從上到下順序。
列  1 :只有結點 20 在此列中。
列  2 :只有結點 7 在此列中。

示例 2:
輸入: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:
輸入:root = [1,2,3,4,6,5,7]
輸出:[[4],[2],[1,5,6],[3],[7]]
解釋:
這個示例實際上與示例 2 完全相同,只是結點 5 和 6 在樹中的位置發生了交換。
因為 5 和 6 的位置仍然相同,所以答案保持不變,仍然按值從小到大排序。

題解思路

返回垂序遍歷的結果,相對於以前的層序遍歷,只不過是換了個方向

層序遍歷是按照每行來橫向便利,所以我們這次要按照列來縱向便利

因為要統計行、列和節點值,所以雜湊表就不太靈活了,我們用陣列加連結串列來代替

正確程式碼

    public List<List<Integer>> verticalTraversal(TreeNode root) {
        List<int[]> nodes = new ArrayList<>();
        dfs(root, 0, 0, nodes);

        // 定義排序方式
        // 按照列號優先,之後是行號,最後是值從小到大
        Collections.sort(nodes, new Comparator<int[]>() {
            @Override
            public int compare(int[] o1, int[] o2) {
                if (o1[0] != o2[0]) {
                    return o1[0] - o2[0];
                } else if (o1[1] != o2[1]) {
                    return o1[1] - o2[1];
                } else {
                    return o1[2] - o2[2];
                }
            }
        });

        // 按列生成返回連結串列
        List<List<Integer>> ans = new ArrayList<>();
        int size = 0;
        int lastcol = Integer.MIN_VALUE;
        for (int[] node : nodes) {
            int col = node[0], row = node[1], value = node[2];
            if (col != lastcol) {
                lastcol = col;
                ans.add(new ArrayList<Integer>());
                size++;
            }
            ans.get(size - 1).add(value);
        }
        return ans;
    }

    // 深度優先遍歷並並儲存每個節點的座標和值
    private void dfs(TreeNode node, int row, int col, List<int[]> nodes) {
        if (node == null) {
            return;
        }
        nodes.add(new int[]{col, row, node.val});
        dfs(node.left, row + 1, col - 1, nodes);
        dfs(node.right, row + 1, col + 1, nodes);
    }

總結

記得有道題也是需要儲存三個值,但是隻有兩個值作為比較條件,用的連結串列搭配雜湊表

本題三個值都是同方式的比較,所以需要用陣列來作為等價的容器,核心在於比較條件的確立

文章如果存在問題或者有更好的題解,歡迎在評論區斧正和評論,各自努力你我最高處見