Deep Dive into Data structures using Javascript - Binary Search Tree

Deep Dive into Data structures using Javascript - Binary Search Tree

Anatomy of a Binary Search Tree, Use cases, in-depth explanation of implementations using both iterative and recursive approaches.

Binary Search Tree (BST as short) is a variation of Binary tree data structure with specific restrictions that allows efficient insertion, deletion and lookup of items. Being a descendant of Binary Tree, it comes with each node having at most two children. Specific restriction is the left child node has a value less than its parent node, while the right child node has a value greater than its parent node.

The tone of this article is assuming you are at least familiar with the concept of Tree data structure and Binary Tree. If that's not the case or you need a quick refreshment, I'd suggest you start from the "Introduction to Trees" article by following the link below, then come back and continue here later:

Deep dive into data structures using Javascript - Introduction to Trees

A Binary Search Tree consists of connected nodes that resemble a hierarchy, where each node has at most two children. The tree starts with a root node, and every node in the tree descends from the root node. There can be only one root in the tree.

Each node in the BST consists of 3 properties:

value: Holds the value / data for the node.

left: Holds a reference (pointer) to the left child node.

right: Holds a reference (pointer) to the right child node.

As mentioned earlier, unique part of Binary Search Tree is the left child node is always having less and the right child node is always having greater value than their parent node. Which also means whenever you add or remove elements, this is something you have to take into account to keep the order of BST staying relevant overall the tree - because this rule applies to every single node inside the BST.

Binary search trees (BSTs) are useful in situations where fast search, insertion, and deletion operations are needed. They are particularly useful when the elements being stored are already sorted or when the elements are constantly changing.

Some real-world examples are:

  • Storing and quickly searching through large amounts of data, such as a database index or financial data like stock prices.

  • Dictionary or spell checker, where words are stored in a BST and can be quickly searched for and inserted.

  • Autocomplete feature, where suggestions are stored in a BST and can be quickly searched for as the user types.

  • Data compression, most frequently occurring elements in the data are represented by the higher levels of the tree and the less frequently occurring elements are represented by the lower levels. This allows for more efficient storage of the data, as the elements that appear more frequently in the data take up less space in the tree.

In general, Binary Search Trees are best suited when the data is static and we want to achieve a good balance between search and insert/delete operations.

What does make Binary Search Tree efficient? Let's see it by taking a look at the Big O of common operations in Binary Search Tree:

As you see, we have 2 different time complexity tables above. Let's start with what do we mean by "Balanced" and "Unbalanced" before we move onto the use cases:

In simple terms, Balanced Binary Search Tree is a tree where the height of left and right subtrees is almost same, and Unbalanced Binary Search Tree is a tree where the height of left and right subtrees is not same, which results in poor performance when searching, inserting, or deleting nodes.

If you take a look at the visual below, you'll see an Unbalanced Binary Search Tree starts to look very similar to a Linked List. And that's the reason why the time complexities on an Unbalanced version is O(n) while the Balanced version have the O(log n):

When does a Binary Search Tree becomes unbalanced? If the items are inserted in an order that creates this skewed / unbalanced order. For example, adding items in an ascending or descending order is a common reason. There are a couple ways to avoid this:

  • Using self balanced BST variations like AVL or Red-Black Tree.

  • Sort the elements in a balanced order before inserting into the BST.

  • Rebalance the BST periodically using a method.

Although AVL or Red-Black Tree implementations of BST adds more complexity to the code, but they provide guaranteed logarithmic O(log n) time complexity for the operations. But for now, we will only focus on the foundational building blocks in this article.

We will be using ES6 Classes to build our Binary Search Tree. In general, implementing non-linear data structures are quite complex - but no worries: we will first go through a couple of steps in detail until we understand the key concepts.

The main building block of a BST is the Node element. We can use a class for it, so we can reuse it whenever we need to create a new Node:

class Node {
  constructor(value) {
    this.value = value
    this.left = null
    this.right = null
  }
}

The Node class has a constructor that takes in a value as a parameter, and initializes the left and right properties to null. We can use these pointers later on to manage referencing other nodes or nullifying them whenever it is needed.

Now that we have our Node class ready, we can build a class for our Binary Search Tree. We know there should be a “root” property in the class as a reference to the root node, which we can place in the constructor:

class BinarySearchTree {
  constructor() {
    this.root = null
  }
}

Before we start with method implementations, I want to point out the importance of understanding the difference between iterative and recursive approaches when working with Binary Search Trees. Both approaches have their own benefits and trade-offs, and it's beneficial to understand which one to use in different scenarios.

An iterative approach uses a loop to traverse through the tree and perform operations. It is generally more efficient in terms of memory usage as it does not require the use of a call stack (as it avoids the overhead of the function call stack). However, it can be more difficult to read and understand for some people and can make the code less readable.

On the other hand, a recursive approach uses function calls to traverse through the tree. This approach is generally simpler to implement, as it follows the natural recursion of the tree structure (side note: Trees are sometimes also referred as recursive data structures). However, it can lead to stack overflow errors when dealing with large trees, and can be less efficient in terms of memory usage.

When working with Binary Search Trees, it's important to consider the size of the tree and the complexity of the operations you'll be performing. For small trees or simple operations, the recursive approach may be more appropriate. However, for large trees or complex operations, the iterative approach may be more efficient and less prone to errors. It's worth noting that JavaScript has a call stack size limit depending on the environment, which means that a recursive function can only be called until it hits the limit before a stack overflow error occurs. Therefore, it is important to keep in mind the size of the tree and the complexity of the operation when deciding which approach to use.

In this article, I will be providing implementation examples for both iterative and recursive approaches for the most common Binary Search Tree operations such as insert, search and delete. This will help you to understand the pros and cons of each approach, and help you make informed decisions when working with BSTs in your own projects.

Now that we have the base building blocks and also understanding of the differences between Iterative and Recursive approaches, we can extend the BinarySearchTree class by introducing methods. Here is the list of methods we are going to implement:

  • insert(value, isRecursive(optional)) - insert a new value into the BST. "isRecursive" parameter is optional, and if set to true, the method will use the "_insertRecursive" helper method to insert the value into the tree. If not provided or set to false, the "_insertIterative" helper method will be used.

  • _insertIterative(value) - helper method that is used to insert a new value into the BST iteratively.

  • _insertRecursive(value) - helper method that is used to insert a new value into the BST recursively.

  • find(value, isRecursive(optional)) - find an element that matches the given value in the BST. "isRecursive" parameter is optional, and if set to true, the method will use the "_findRecursive" helper method to find the value. If not provided or set to false, the "_findIterative" helper method will be used.

  • _findIterative(value) - helper method that is used to find an element that matches the given value in the BST iteratively.

  • _findRecursive(value) - helper method that is used to find an element that matches the given value in the BST recursively.

  • delete(value, isRecursive(optional)) - remove a node that matches the given value from the BST. "isRecursive" parameter is optional, and if set to true, the method will use the "_deleteRecursive" helper method to remove the node. If not provided or set to false, the "_deleteIterative" helper method will be used.

  • _deleteIterative(value) - helper method that is used to remove a node that matches the given value from the BST iteratively.

  • _deleteRecursive(value) - helper method that is used to remove a node that matches the given value from the BST recursively.

  • showAsLevelOrdered - display the BST as a level-order multi-dimensional array, uses "level order traversal" under the hood. The final output array will represent the tree in level-order, with each sub-array representing a level of the tree.

Now you might be asking, what does "Level order traversal" means that I mentioned in showAsLevelOrdered description? It is a subject related to Tree Traversals which needs it's own article. To keep things simple I won’t be diving deep into Tree traversal details here - but rather give a quick overview about it.

Traversals are a way to traverse through all the nodes in a tree data structure, visiting each node in a specific order. "Level order traversal" is one of them, which is also known as "Breadth-first traversal". With level order, nodes are visited level by level, starting from the root. This type of traversal is useful for displaying the tree in a readable format, as it shows the tree's structure clearly. Here is a rough sketch of how the output looks like:

One thing I'd like to mention is, it's completely normal to not fully understand Binary Search Trees at first glance. It's quite common whenever you are dealing with any non-linear data structure. The rule of thumb is, if you give enough time it will all make sense!

I hope this article helped you to understand what Binary Search Tree is and how it works! I’d like to encourage you to experiment with the implementation below in your favorite code editor. When you are experimenting with Binary Search Tree, I highly recommend using the Visualgo on the side. It has a great visualition that can show you how everything works visually.

Here is the link to Binary Search Tree visualization on Visualgo: https://visualgo.net/en/bst

Thanks for reading!

class Node {
  constructor(value) {
    this.value = value
    this.left = null
    this.right = null
  }
}


class BinarySearchTree {
  constructor() {
    // initialize the root
    this.root = null
  }

  // handles "insertion" with both variations: iterative and recursive
  // "isRecursive" is an optional parameter, default handler is iterative
  insert(value, isRecursive) {
    if (!isRecursive) return this._insertIterative(value)
    return this._insertRecursive(value)
  }

  // handles "find" with both variations: iterative and recursive
  // "isRecursive" is an optional parameter, default handler is iterative
  find(value, isRecursive) {
    if (!isRecursive) return this._findIterative(value)
    return this._findRecursive(value)
  }

  // handles "delete" with both variations: iterative and recursive
  // "isRecursive" is an optional parameter, default handler is iterative
  delete(value, isRecursive) {
    if (!isRecursive) return this._deleteIterative(value)
    return this._deleteRecursive(value)
  }

  // Iterative insertion
  _insertIterative(value) {
    // Create a new node with the value to be inserted
    const newNode = new Node(value)

    // If the tree is empty, make the new node as the root and early exit
    if (this.root === null) {
      return this.root = newNode
    }

    // Iterative insert method to be called if the root is not empty:
    const iterativeInsert = (currentNode, value) => {
      while (true) {
        // If the value already exists in the tree,
        // throw an error or silently fail:
        if (value === currentNode.value) {
          // OPTIONAL: replace the below line with return 
          // if you want it to throw an error on duplicates
          // throw new Error("Duplicate value not allowed")
          return
        }

        // Check if the input value is less than the current node's value
        if (value < currentNode.value) {
          // If the left child is null, insert the new node as the left child
          if (currentNode.left === null) {
            return currentNode.left = newNode
          }
          // If the left child is not null, move the current node to the left child
          currentNode = currentNode.left
        } else {
            // Check if the input value is greater than
            // or equal to the current node's value
            if (currentNode.right === null) {
              // If the right child is null, insert the new node as the right child
              return currentNode.right = newNode
            } else {
              // If the right child is not null, move the current node to the right child
              currentNode = currentNode.right
            }
        }
      }
    }

    // Call the inner method with root node
    // as the starting point, and value as target:
    iterativeInsert(this.root, value)
  }

  // Recursive insertion
  _insertRecursive(value) {
    // Create a new node with the value to be inserted
    const newNode = new Node(value)

    // if the tree is empty, insert the new node as the root and early exit
    if (this.root === null) {
      return this.root = newNode
    }

    // Recursive insert method to be called if the root is not empty:
    const recursiveInsert = (currentNode, value) => {
        // If the value already exists in the tree, throw an error or silently fail:
        if (value === currentNode.value) {
          // OPTIONAL: replace the below line with return 
          // if you want it to throw an error on duplicates
          // throw new Error("Duplicate value not allowed")
          return
        }

        // if the value is greater than the current node's value, go right
        if (value > currentNode.value) {
            // if there is no right child, insert the new node here
            if (currentNode.right === null) {
              currentNode.right = newNode
              return
            } else {
                // otherwise, recursively call insert on the right child
                recursiveInsert(currentNode.right, value)
            }
        } else {
            // if the value is less than the current node's value, go left
            if (currentNode.left === null) {
                currentNode.left = newNode
                return
            } else {
                // otherwise, recursively call insert on the left child
                recursiveInsert(currentNode.left, value)
            }
        }
    }

    // Call the inner method with root node
    // as the starting point, and value as target:
    recursiveInsert(this.root, value)
  }

  // Iterative search
  _findIterative(value, currentNode = this.root) {
    // Start at the root of the tree (or a passed in node)
    while (currentNode) {
      // Check if the current node's value is equal to the search value
      if (currentNode.value === value) {
        return currentNode.value; // If found, return the value
      }
      // If the search value is less than the current node's value,
      // move to the left child
      if (value < currentNode.value) {
        currentNode = currentNode.left;
      } else {
        // If the search value is greater than the current node's value,
        // move to the right child
        currentNode = currentNode.right;
      }
    }
    // If the value is not found, return null
    return console.log('Given value not found in the tree:', value)
  }

  // Recursive search
  _findRecursive(value, currentNode = this.root) {    
    // base case: if current node is null, value is not in the tree
    if (currentNode === null) {
      return console.log('Given value not found in the tree:', value)
    }

    // if current node value is equal to the given value, return value
    if (currentNode.value === value) {
      return currentNode.value
    }

    // if given value is less than current node value, search left subtree
    if (value < currentNode.value) {
      return this._findRecursive(value, currentNode.left)
    }

    // if given value is greater than current node value, search right subtree
    if (value > currentNode.value) {
      return this._findRecursive(value, currentNode.right)
    }
  }

  // Iterative delete
  _deleteIterative(value) {
    // Start at the root
    let currentNode = this.root
    // At this point parentNode is null, since root has no parent
    let parentNode = null

    // STEP 1 - Find the targetNode and it's parentNode to be deleted:
    while (currentNode !== null && currentNode.value !== value) {
      /*
      Here, the currentNode is set as the parentNode 
      before moving to the left or right child node 
      depending on the value being searched for.
      This is so that the parentNode variable always 
      holds the parent of the currentNode.

      When the loop exits, the currentNode variable holds the target node
      that needs to be deleted and "parentNode" variable holds its parent.
      */
      parentNode = currentNode

      // if the value is less than the current node's value, move to the left child
      if (value < currentNode.value) {
        currentNode = currentNode.left
      // if the value is greater than the current node's value, move to the right child
      } else {
        currentNode = currentNode.right
      }
    }

    // STEP 2 - If the currentNode is null, the value was not found in the tree
    if (currentNode === null) {
      console.log('Given value not found in the tree:', value)
      return
    }

    // STEP 3 - Handle the 3 deletion cases: 
    // leaf node (node with 0 children), 
    // node with 1 children, node with 2 children

    // CASE 1: LEAF NODE
    // If the current node has no children, it is a leaf node. 
    if (currentNode.left === null && currentNode.right === null) {
      /*
      In this case, if the parent node is null 
      (which means the current node is the root), 
      the root is set to null.
      */
      if (parentNode === null) {
        this.root = null        
      } else {
      /*
      If the parent node is not null, the left or right reference
      of the parent node is set to null, depending on 
      if the current node is the left or right child of the parent node.
      */
        if (parentNode.left === currentNode) {
          parentNode.left = null
        } else {
          parentNode.right = null
        }
      }
    }

    // CASE 2: NODE WITH SINGLE CHILD
    // If the current node has 1 children at the left or right side

    // ONLY HAS CHILDREN AT RIGHT SIDE, LEFT IS NULL
    else if (currentNode.left === null) {
      /*
      In this case, if the parent node is null 
      (which means the current node is the root), 
      the root is set to the current node's right child.
      */
      if (parentNode === null) {
        this.root = currentNode.right
      } else {
        /*
        If the parent node is not null, the left or right
        reference of the parent node is set to 
        the current node's right child, depending on 
        if the current node is the left or right child of the parent node.
        */
        if (parentNode.left === currentNode) {
          parentNode.left = currentNode.right
        } else {
          parentNode.right = currentNode.right
        }
      }
    // ONLY HAS CHILDREN AT LEFT SIDE, RIGHT IS NULL
      /*
      Same as the above block of code, but the children at the opposite side
      */
    } else if (currentNode.right === null) {
      if (parentNode === null) {
        this.root = currentNode.left
      } else {
        if (parentNode.left === currentNode) {
          parentNode.left = currentNode.left
        } else {
          parentNode.right = currentNode.left
        }
      }
    }

    // CASE 3: NODE WITH TWO CHILDREN
    // If the current node has 2 children (left & right)
    else {
      /*
      If the current node has two children, 
      we need to find the in-order successor of the current node.

      You can go 2 ways to find the inorder successor: either smallest value 
      in right subtree, or largest element in left subtree. In this method 
      we will use the first: finding the smallest value in right subtree.

      The in-order successor is the node with the smallest value 
      that is greater than the current node's value.

      We start by initializing the inorderSuccessor variable to the 
      current node's right child and inorderSuccessorParent to the current node.

      Then, we iterate through the right child's left subtree 
      until we find the node with the smallest value.
      */
      let inorderSuccessor = currentNode.right
      let inorderSuccessorParent = currentNode

      while (inorderSuccessor.left !== null) {
        inorderSuccessorParent = inorderSuccessor
        inorderSuccessor = inorderSuccessor.left
      }

      currentNode.value = inorderSuccessor.value
      /*
      Once we find the in-order successor, 
      we replace the current node's value with the in-order successor's value.

      Then, we remove the in-order successor from the tree 
      by setting its parent's left or right pointer to its right child.
      (since the in-order successor is guaranteed to have 
      no left children, we can simply remove it by linking 
      its parent to its right child).

      Finally, the method returns and the tree is updated with 
      the desired value removed and its in-order successor taking its place.
      */
      if (inorderSuccessorParent === currentNode) {
        inorderSuccessorParent.right = inorderSuccessor.right
      } else {
        inorderSuccessorParent.left = inorderSuccessor.right
      }
    }
  }

  // Recursive delete
  _deleteRecursive(value) {
    // Recursive delete method to be called
    const recursiveDelete = (currentNode, value) => {
      // Base case: if the node is null, the value is not in the tree
      if (currentNode === null) {
        console.log('Given value not found in the tree:', value)
        return
      }
      // if the value is less than the node's value, 
      // then it lies in left subtree
      if (value < currentNode.value) {
        currentNode.left = recursiveDelete(currentNode.left, value)
      }
      // if the value is more than the node's value, 
      // then it lies in right subtree
      else if (value > currentNode.value) {
        currentNode.right = recursiveDelete(currentNode.right, value)
      }
      // if value is same as node's value, then this is the node
      // to be deleted
      else {
        // node with only one child or no child
        if (currentNode.left === null) {
          return currentNode.right
        } else if (currentNode.right === null) {
          return currentNode.left
        }

        // node with two children
        // get the inorder successor (smallest in the right subtree)
        let inorderSuccessor = currentNode.right
        while (inorderSuccessor.left !== null) {
          inorderSuccessor = inorderSuccessor.left
        }
        // copy the inorder successor's content to this node
        currentNode.value = inorderSuccessor.value

        // delete the inorder successor
        currentNode.right = recursiveDelete(currentNode.right, inorderSuccessor.value)
      }
      return currentNode
    }

    /* 
    And as usual, we always start the traversal from 
    the top of tree "this.root". recursiveDelete helper method 
    will return the complete updated tree after removal, 
    so we simply point the root to the updated tree:
    */
    this.root = recursiveDelete(this.root, value)
  }

  // Displays an array that will represent the tree
  // in level-order, with each sub-array representing a level of the tree.
  showAsLevelOrdered() {
    // Create an empty array to store the traversed nodes
    const temp = []
    // Create an array to keep track of the current level of nodes
    const queue = []

    // If the tree has a root, add it to the queue
    if (this.root) {
      queue.push(this.root)      
    }

    // Keep traversing the tree while there are nodes in the queue
    while (queue.length) {
      // Create an array to store the nodes of the current level
      const subTemp = []
      // Store the number of nodes in the current level
      const len = queue.length

      // Iterate through the current level of nodes
      for (let i = 0; i < len; i += 1) {
        // Dequeue the first node in the queue
        const node = queue.shift()
        // Push the node's value to the subTemp array
        subTemp.push(node.value)
        // If the node has a left child, add it to the queue
        if (node.left) {
          queue.push(node.left)
        }
        // If the node has a right child, add it to the queue
        if (node.right) {
          queue.push(node.right)
        }
      }

      // Push the subTemp array to the temp array
      temp.push(subTemp)
    }
    // Return the final temp array
    return temp
  }
}