Print Boundary View In Binary Tree

Given a Binary tree, the task is to Print Boundary View in Binary Tree, visiting the boundary nodes in an anticlockwise direction starting from the root.

Print Boundary View In Binary Tree

Code Flow for Print Boundary View In Binary Tree

  • Start at the root.

  • Add the root to the result if it's not a leaf.

  • Traverse and add left boundary nodes, excluding leaf nodes.

  • Traverse and add bottom boundary nodes, ie, all leaf nodes.

  • Traverse and add right boundary nodes, excluding leaf nodes, which are added in reverse order.

  • Return the list of boundary nodes.


Let's see the code now

package org.wesome.dsalgo;

import lombok.Data;

import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

@Data
class Node {
    int data;
    Node left;
    Node right;

    Node(int val) {
        data = val;
    }
}

public class BinaryTree {
    /*  check if a node is a leaf Node, i.e., if it has left and right child node, then this node will not be a boundary    */
    boolean isLeafNode(Node root) {
        var bool = Objects.isNull(root.left) && Objects.isNull(root.right);
        System.out.println("BinaryTree.isLeafNode:- " + root.data + " is leaf node " + bool);
        return bool;
    }

    /*  add Left Leaf nodes excluding Bottom leaf Node, because those will be part of the bottom boundary */
    void addLeftLeafNode(Node root, List<Integer> result) {
        Node currNode = root.left;
        while (Objects.nonNull(currNode)) {
            /*  if the current node is not a leaf node, i.e., it has a child node, then add its value into the list  */
            if (!isLeafNode(currNode)) {
                result.add(currNode.data);
            }
            /*   if the current node left exists, then move to next left else move to right    */
            if (Objects.nonNull(currNode.left)) {
                currNode = currNode.left;
            } else {
                currNode = currNode.right;
            }
        }
    }

    /*  add right Leaf nodes excluding Bottom leaf Node in reverse order, because those will be part of the bottom boundary */
    void addRightLeafNode(Node root, List<Integer> result) {
        Node currNode = root.right;
        List<Integer> tempList = new ArrayList<>();
        while (Objects.nonNull(currNode)) {
            /*   if the current node is not a leaf node, i.e., it has child nodes, then add it to the temp list for further iteration   */
            if (!isLeafNode(currNode)) {
                tempList.add(currNode.data);
            }
            /*   if the current node right exists, then move to next right else move to left    */
            if (Objects.nonNull(currNode.right)) {
                currNode = currNode.right;
            } else {
                currNode = currNode.left;
            }
        }

        /*  the right leaf nodes are added in top to down or clock wise direction, but the result has to be bottom to up or anticlockwise wise direction, hence reverse the temp list   */
        result.addAll(tempList.reversed());

    }
    /*  traverse in a preorder manner, i.e., left-> root-> right, but ignore root since it has child hence it's not boundary, add bottom Leaf nodes */

    void addBottomLeafNode(Node root, List<Integer> result) {
        /*  if the node has no left and right child, then its bottom leaf node */
        if (isLeafNode(root)) {
            System.out.println("BinaryTree.addBottomLeafNode:- root = " + root.data);
            result.add(root.data);
            return;
        }
        /*  if the node has a left and right child, then it's not a bottom leaf node, so keep iterating left and right node */
        if (Objects.nonNull(root.left)) {
            addBottomLeafNode(root.left, result);
        }
        if (Objects.nonNull(root.right)) {
            addBottomLeafNode(root.right, result);
        }
    }


    List<Integer> printBinaryTreeLeaf(Node root) {
        List<Integer> list = new ArrayList<>();
        if (Objects.isNull(root)) {
            System.out.println("BinaryTree.printBinaryTreeLeaf root is null");
            return list;
        }
        /*  if root is not a leaf node, i.e., it has child nodes, add in list  */
        if (!isLeafNode(root)) {
            System.out.println("BinaryTree.printBinaryTreeLeaf root is not a leaf node");
            list.add(root.data);
        }

        /*  add left leaf nodes in the list */
        addLeftLeafNode(root, list);

        /*  add bottom leaf nodes in the list */
        addBottomLeafNode(root, list);
        /*  add left leaf nodes in the list */
        addRightLeafNode(root, list);
        return list;
    }
}
package org.wesome.dsalgo;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

import java.util.List;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertTrue;

public class BinaryTreeTest {

    private BinaryTree binaryTree;

    @BeforeEach
    void setUp() {
        binaryTree = new BinaryTree();
    }

    // Test case 1: Empty Tree
    @Test
    void testEmptyTree() {
        Node root = null;
        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertTrue(result.isEmpty(), "The result should be empty for an empty tree.");
    }

    // Test case 2: Tree with only one node (Leaf node)
    @Test
    void testSingleNode() {
        Node root = new Node(1);
        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1), result, "The result should contain only the single root node.");
    }

    // Test case 3: Tree with two nodes (One parent and one child)
    @Test
    void testTwoNodes() {
        Node root = new Node(1);
        root.left = new Node(2); // left child
        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 2), result, "The result should contain the boundary nodes: root and the leaf node.");
    }

    // Test case 4: Simple tree with left and right subtrees
    @Test
    void testSimpleTree() {
        // Constructing the following tree:
        //         1
        //       /   \
        //      2     3
        //       \   /
        //        4 5
        Node root = new Node(1);
        root.left = new Node(2);
        root.left.right = new Node(4);
        root.right = new Node(3);
        root.right.left = new Node(5);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 2, 4, 5, 3), result, "The result should contain the boundary nodes in anticlockwise order.");
    }

    // Test case 5: Tree with only left children (Linear Tree)
    @Test
    void testLeftLinearTree() {
        // Constructing the following tree:
        //         1
        //        /
        //       2
        //      /
        //     3
        Node root = new Node(1);
        root.left = new Node(2);
        root.left.left = new Node(3);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 2, 3), result, "The result should contain the boundary nodes in anticlockwise order.");
    }

    // Test case 6: Tree with only right children (Linear Tree)
    @Test
    void testRightLinearTree() {
        // Constructing the following tree:
        //         1
        //          \
        //           2
        //            \
        //             3
        Node root = new Node(1);
        root.right = new Node(2);
        root.right.right = new Node(3);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 3, 2), result, "The result should contain the boundary nodes in anticlockwise order.");
    }

    // Test case 7: Tree with both left and right children with a larger structure
    @Test
    void testLargeTree() {
        // Constructing the following tree:
        //          1
        //       /     \
        //      2       3
        //     / \     / \
        //    4   5   6   7
        Node root = new Node(1);
        root.left = new Node(2);
        root.right = new Node(3);
        root.left.left = new Node(4);
        root.left.right = new Node(5);
        root.right.left = new Node(6);
        root.right.right = new Node(7);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 2, 4, 5, 6, 7, 3), result, "The result should contain the boundary nodes in anticlockwise order.");
    }

    // Test case 8: Tree with all leaf nodes on the right side
    @Test
    void testAllLeafNodesRight() {
        // Constructing the following tree:
        //          1
        //           \
        //            2
        //             \
        //              3
        //               \
        //                4
        Node root = new Node(1);
        root.right = new Node(2);
        root.right.right = new Node(3);
        root.right.right.right = new Node(4);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 4, 3, 2), result, "The result should contain the boundary nodes in anticlockwise order.");
    }

    // Test case 9: Tree with duplicate values
    @Test
    void testTreeWithDuplicateValues() {
        // Constructing the following tree:
        //          1
        //       /     \
        //      1       1
        //     / \     / \
        //    1   1   1   1
        Node root = new Node(1);
        root.left = new Node(1);
        root.right = new Node(1);
        root.left.left = new Node(1);
        root.left.right = new Node(1);
        root.right.left = new Node(1);
        root.right.right = new Node(1);

        List<Integer> result = binaryTree.printBinaryTreeLeaf(root);
        assertEquals(List.of(1, 1, 1, 1, 1, 1, 1), result, "The result should contain all boundary nodes in anticlockwise order.");
    }


    @Test
    void testTreeWithCustomValues() {
        // Creating a sample binary tree
        Node root = new Node(1);
        root.left = new Node(2);
        root.right = new Node(3);
        root.left.left = new Node(4);
        root.left.right = new Node(5);
        root.right.left = new Node(6);
        root.right.right = new Node(7);
        assertEquals(List.of(1, 2, 4, 5, 6, 7, 3), binaryTree.printBinaryTreeLeaf(root), "The result should contain the root, left boundary nodes and right boundary nodes in anticlockwise order.");
    }
}
plugins {
    id("java")
    id("io.freefair.lombok") version "8.13"
}

group = "org.wesome.dsalgo"
version = "1.0-SNAPSHOT"

repositories {
    mavenCentral()
}

dependencies {
    testImplementation(platform("org.junit:junit-bom:5.10.0"))
    testImplementation("org.junit.jupiter:junit-jupiter")
}

tasks.test {
    useJUnitPlatform()
}

Complexity Analysis

Time Complexity

  1. Time Complexity: O(n), where n is the number of nodes in the tree.

Space Complexity

  1. Space Complexity: O(n), where n is the number of nodes in the tree.

follow us on