Solution: Vertical Order Traversal of a Binary Tree
Let's solve the Vertical Order Traversal of a Binary Tree problem using the Tree Breadth-first Search pattern.
Statement
Find the vertical order traversal of a binary tree when the root of the binary tree is given. In other words, return the values of the nodes from top to bottom in each column, column by column from left to right. If there is more than one node in the same column and row, return the values from left to right.
Constraints:
- The number of nodes in the tree is in the range
[1, 500]
. -
Node.data
Solution
So far, you’ve probably brainstormed some approaches and have an idea of how to solve this problem. Let’s explore some of these approaches and figure out which one to follow based on considerations such as time complexity and any implementation constraints.
Naive approach
The naive approach would be traversing the tree to get the maximum and minimum horizontal distance of the nodes from the root. Once we have these numbers, we can traverse over the tree again for each distance in the range [maximum, minimum]
and return the nodes respectively.
The time complexity of the above algorithm is , where is the number of nodes. The space complexity is .
Optimized approach using tree breadth-first search
Since we’re dealing with a binary tree, the approach that comes to mind is either depth-first search (DFS) or breadth-first search (BFS), which are both viable options. We will select BFS to solve this problem. Using BFS, we can assign the value to the left child and the to the right child.
The intuition behind using BFS is that it naturally guarantees that the nodes will be visited level by level. This ensures that the nodes at the top will be visited first, and the nodes at the bottom will be visited last. To traverse the tree from left to right, we need to maintain a hash map that contains the list of all the nodes in a specific column. We’ll denote these columns by their index, where the column of the root will be index 0
, the columns on the left side of the root will have a negative index, and the columns on the right side of the root will have a positive index. The index of each column will be a key.
We also need to store the value of the minimum and maximum indexes in two variables so that we can iterate over the indexes to get the nodes in the right order.
Let’s go over the algorithm in detail below:
-
We need to create a hash map called
nodeList
in which the key will be the index of the column and the value will be the list of nodes in that column. -
We initialize two variables to store the minimum and maximum values of the index.
-
During BFS, we also maintain a queue to keep track of the order of nodes that need to be visited. We’ll initialize the queue by adding the root to it along with the column index of the root, which is
0
. -
Next, we carry out BFS using a loop until the queue is empty. We iterate over the queue elements and pop them. The values in the queue consist of a node and the index of the column.
-
If the queue contains a node that is not NULL, we add the value of the node to
nodeList
along with the index of the column. We also keep updating the minimum and maximum values of the index. Then, we append the children of this node to the queue with their index values. The left child has an index value of the current index , and the right child has a value of the current index . -
At the end of all the iterations, the
nodeList
hash map contains the values of all the nodes in all the columns, but the values of each column are not in the correct order. -
We’ll iterate over the index range using the variables that stored the minimum and maximum indexes. Finally, we return the values of the nodes in the correct order.
Let’s look at the code for this solution below:
class Solution {public static List<List<Integer>> verticalOrder(TreeNode<Integer> root) {List<List<Integer>> output = new ArrayList<List<Integer>> ();if (root == null) {return output;}Map<Integer, ArrayList<Integer>> nodeList = new HashMap<Integer, ArrayList<Integer>> ();Queue<Map.Entry<TreeNode<Integer>, Integer>> queue = new ArrayDeque<Map.Entry<TreeNode<Integer>, Integer>>();int column = 0;queue.offer(new AbstractMap.SimpleEntry<TreeNode<Integer>, Integer>(root, column));int minColumn = 0;int maxIndex = 0;while (!queue.isEmpty()) {Map.Entry<TreeNode<Integer>, Integer> p = queue.poll();root = p.getKey();column = p.getValue();if (root != null) {if (!nodeList.containsKey(column)) {nodeList.put(column, new ArrayList<Integer> ());}nodeList.get(column).add(root.data);minColumn = Math.min(minColumn, column);maxIndex = Math.max(maxIndex, column);queue.offer(new AbstractMap.SimpleEntry<TreeNode<Integer>, Integer>(root.left, column - 1));queue.offer(new AbstractMap.SimpleEntry<TreeNode<Integer>, Integer>(root.right, column + 1));}}for (int i = minColumn; i<maxIndex + 1; ++i) {output.add(nodeList.get(i));}return output;}// Driver codepublic static void main(String[] args) {List<List<TreeNode<Integer>>> listOfTrees = Arrays.asList(Arrays.asList(new TreeNode<Integer>(100), new TreeNode<Integer>(50), new TreeNode<Integer>(200), new TreeNode<Integer>(25), new TreeNode<Integer>(75), new TreeNode<Integer>(300), new TreeNode<Integer>(10), new TreeNode<Integer>(350), new TreeNode<Integer>(15)),Arrays.asList(new TreeNode<Integer>(20), new TreeNode<Integer>(40), new TreeNode<Integer>(50), new TreeNode<Integer>(90), new TreeNode<Integer>(67), new TreeNode<Integer>(94)),Arrays.asList(new TreeNode<Integer>(-10), new TreeNode<Integer>(-23), new TreeNode<Integer>(45), new TreeNode<Integer>(25), new TreeNode<Integer>(46)),Arrays.asList(new TreeNode<Integer>(9), new TreeNode<Integer>(7), null, null, new TreeNode<Integer>(1), new TreeNode<Integer>(8), new TreeNode<Integer>(10), null, new TreeNode<Integer>(12)),Arrays.asList(new TreeNode<Integer>(3), new TreeNode<Integer>(2), new TreeNode<Integer>(3), null, new TreeNode<Integer>(3), null, new TreeNode<Integer>(1)));List<BinaryTree<Integer>> inputTrees = new ArrayList<>();for (List<TreeNode<Integer>> ListOfNodes : listOfTrees) {BinaryTree<Integer> tree = new BinaryTree<>(ListOfNodes);inputTrees.add(tree);}int x = 1;for (BinaryTree<Integer> tree : inputTrees) {System.out.println(x + ".\tInput Tree:");Print.displayTree(tree.root);x++;System.out.println("\n\tVertical order: " + verticalOrder(tree.root));System.out.println(new String(new char[100]).replace('\0', '-'));}}}
Solution summary
To recap, the solution to this problem can be divided into the following steps:
- Traverse the tree level by level, starting from the root.
- Maintain a hash map that contains the list of all the nodes in a specific column.
- Store the value of the minimum and maximum indexes in two variables so we can iterate over the indexes to get the nodes in the right order.
Time complexity
We are using BFS, which goes through each node precisely once and has a time complexity of , where is the number of nodes in the binary tree.
Space complexity
In this solution, there are two data structures, a hash map and a queue, that play a significant role in the overall space complexity.
The hash map stores the nodes’ values against the column indexes. In the worst-case scenario, every node will be in a different column. That means its space complexity will be , where is the number of nodes in the binary tree.
At any time, the queue will contain a maximum of nodes of two levels, and the maximum a level can contain is . So, in the worst-case scenario, the space complexity will be = .
Therefore, the overall space complexity of this solution is .
Level up your interview prep. Join Educative to access 70+ hands-on prep courses.