Binary Search Trees#
Binary Tree#
Overview
A binary tree is a hierarchical data structure consisting of nodes.
Each node contains a value and references to its left and right children.
The left child is smaller than the parent, while the right child is larger.
Implementing binary trees#
- Node
\(data\)
\(left\) child
\(right\) child
data:image/s3,"s3://crabby-images/a457e/a457e45cd96be07789899070893df88aeee3e16e" alt="../../_images/15_s7.png"
\(tree = \{a, b, c, d, e, f\}\)
\(k\)-ary trees#
data:image/s3,"s3://crabby-images/75aaf/75aaf1d13c98a39efc9d5bc6fcbdedc68d5eccda" alt="https://media.geeksforgeeks.org/wp-content/cdn-uploads/20201102101430/Tree2.jpg"
Fig. 14 Every node has between \(0\) and \(k\) children#
data:image/s3,"s3://crabby-images/511ab/511ab3670483e9fdca6d08945bca910195aa4aa8" alt="https://miro.medium.com/v2/resize:fit:1400/format:webp/1*CMGFtehu01ZEBgzHG71sMg.png"
data:image/s3,"s3://crabby-images/a7282/a72826660be3d50fbbedf9ba2bd08648180c85a0" alt="https://miro.medium.com/max/1400/1*fh2By4u-SxTlt6u2xHqnCg.png"
Fig. 15
Full Binary Tree is a Binary Tree in which every node has 0 or 2 children.
– towardsdatascience.com#
data:image/s3,"s3://crabby-images/4cfa6/4cfa68bd3c83ce1d254e5104965853031b99e8b3" alt="https://miro.medium.com/max/1400/1*M1qfRR59TR9-i4pmI-_Clg.png"
Fig. 16
Complete Binary Tree has all levels completely filled with nodes except the last level and in the last level, all the nodes are as left side as possible.
– towardsdatascience.com#
data:image/s3,"s3://crabby-images/2735e/2735ea1ef70add8b0945edc99c5d0bcecd0c100c" alt="https://miro.medium.com/max/1400/1*EgcvwUHXnmdOpbHQwgCknA.png"
Fig. 17
Perfect Binary Tree is a Binary Tree in which all internal nodes have 2 children and all the leaf nodes are at the same depth or same level.
– towardsdatascience.com#
data:image/s3,"s3://crabby-images/d84d0/d84d00f49e5ef0bec506edfb1d91d17d6d4f1777" alt="https://miro.medium.com/max/1400/1*m5BjLJeSrSGH4US-QXj4aA.png"
Fig. 18
Degenerate Binary Tree is a Binary Tree where every parent node has only one child node.
– towardsdatascience.com#
data:image/s3,"s3://crabby-images/2aa1a/2aa1aaddd39467429898a5be158d942f2adbfa20" alt="https://miro.medium.com/max/1400/1*jSq-xjEZYytNDIBpZNQC2w.png"
Fig. 19
Balanced Binary Tree is a Binary tree in which height of the left and the right sub-trees of every node may differ by at most 1.
– towardsdatascience.com#
Binary Search Trees#
Understanding BSTs
data:image/s3,"s3://crabby-images/1c317/1c317e3dbaaee8e963cdc66ab7343cf891201cb6" alt="https://blog.penjee.com/wp-content/uploads/2015/11/binary-search-tree-sorted-array-animation.gif"
Each node in a binary tree can have at most two children.
This means that a node can have either zero, one, or two children.
The left child of a node is typically smaller than the node, while the right child is larger (or equal) than the node.
This rule is often associated with binary search trees (BSTs), where the values are arranged in a specific order for efficient searching and retrieval.
The order of children matters.
In a binary tree, the left child is typically referenced before the right child.
The order of children affects the tree’s traversal and interpretation.
Nodes are connected by edges.
The connections between nodes are represented by edges in a binary tree.
Each node (except the root) has exactly one incoming edge from its parent.
The height of a binary tree.
The height of a binary tree is the maximum number of edges from the root to a leaf node.
It represents the longest path from the root to any leaf node in the tree.
The depth (level) of a node.
The depth (or level) of a node in a binary tree represents the number of edges from the root to that node.
The root node is at depth 0, its children are at depth 1, and so on.
The number of nodes in a binary tree.
The total number of nodes in a binary tree can vary.
The minimum number of nodes in a binary tree is 0, while the maximum number depends on the height of the tree.
Simplified is bst
A BST has symmetric order
each node \(x\) in a BST has a key
for all nodes \(y\) in the left subtree of \(x\),
for all nodes \(y\) in the right subtree of \(x\),
data:image/s3,"s3://crabby-images/13e23/13e236ed507286564073d88a339365134e386ab9" alt="https://i.pinimg.com/originals/4e/5a/00/4e5a00c98d5b46ed4cbf224ac11dc115.png"
(**) assume that the keys of a BST are pairwise distinct
data:image/s3,"s3://crabby-images/6d5c1/6d5c13a9ca22dac0ffbfc0c13fde5147d176e6d8" alt="https://res.cloudinary.com/practicaldev/image/fetch/s--od-naD9n--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://miro.medium.com/max/975/1*PWJiwTxRdQy8A_Y0hAv5Eg.png"
BST Classes#
1class BSTNode {
2
3 private:
4 int data;
5 BSTNode *left;
6 BSTNode *right;
7
8 public:
9 BSTNode(int d);
10 ~BSTNode();
11
12 friend class BSTree;
13
14};
1class BSTree{
2
3 private:
4 BSTNode *root;
5 void destroy(BSTNode *p);
6
7 public:
8 BSTree();
9 ~BSTree();
10 void insert(int d);
11 void remove(int d);
12 BSTNode *search(int d);
13
14};
search
#
Start at root node
If the search key:
a. matches the current node’s key
* then found
b. If search key \(>\) current node’s key
* search on \(right\) child
c. If search key is less than current node’s
* search on \(left\) childStop when current node is NULL (not found)
data:image/s3,"s3://crabby-images/cab02/cab02116d92e3b282d838503c56a281f87202f71" alt="https://www.happycoders.eu/wp-content/uploads/2021/06/binary-search-tree-insertion-800x467.png"
\(Search\ for\ \ 8\)
visualize
try it
insert
#
To insert a new value into a BST:
Start at the root node.
If the value is smaller, go to the left subtree; if larger, go to the right subtree.
Repeat steps 2 until an empty spot is found.
Insert the new node in the empty spot.
visualize
try it
remove
#
To delete a node from a BST:
Find the node to delete.
If the node has no children, simply remove it.
If the node has one child, replace it with its child.
If the node has two children, find the node’s successor (smallest value in the right subtree) or predecessor (largest value in the left subtree).
Replace the node with its successor/predecessor and delete the successor/predecessor from its original position.
Fig. 20 \(remove : leaf\)#
Fig. 21 \(remove:\ with\ 1\ child\)#
Fig. 22 \(remove:\ with\ 2\ children\)#
visualize
try it
Traversals#
Implications#
Traversal of a tree
T
is a systematic way of accessing, or “visiting,” all the positions ofT
. The specific action associated with the “visit” of a positionp
depends on the application of this traversal, and could involve anything from incrementing a counter to performing some complex computation forp
.
data:image/s3,"s3://crabby-images/758eb/758ebf9de78c071df71a2b3ceba7a37d4a4d6ee9" alt="https://www.bogotobogo.com/GoLang/images/BinarySearchTree_BST/print_functions.png"
animation
data:image/s3,"s3://crabby-images/0d423/0d42348c26c8bf13713bf4ad9198da09620e4d4c" alt="https://miro.medium.com/v2/resize:fit:1000/1*UGoV21qO6N8JED-ozsbXWw.gif"
try it
animation
data:image/s3,"s3://crabby-images/5cea4/5cea43e95018e4dc64e8d83a8d79adc2b9d1e05d" alt="https://miro.medium.com/v2/resize:fit:1000/1*UGrzA4qtLCaaCiNAKZyj_w.gif"
try it
How would we:
Destroy a binary tree
Post Order
data:image/s3,"s3://crabby-images/5cea4/5cea43e95018e4dc64e8d83a8d79adc2b9d1e05d" alt="https://miro.medium.com/v2/resize:fit:1000/1*UGrzA4qtLCaaCiNAKZyj_w.gif"
Fig. 26
Post Order#
- Why?
It’s the only traversal that visits all the subtree nodes before visiting the root. All child nodes must be removed before we can delete the root.
Print all elements ascending order
In Order
data:image/s3,"s3://crabby-images/03ebe/03ebe631adfc2520cce8ce59424b651f5fd5d7f6" alt="https://miro.medium.com/v2/resize:fit:1000/1*bxQlukgMC9cGv_MFUllX2Q.gif"
- Why?
to output in ascending order, we would need to start with the lowest value sequentially and work our way to the largest; as BST’s are built in sequential order, \(in-order\) fits best
best-case |
worst-case |
average-case |
|
---|---|---|---|
search |
|||
insert |
|||
remove |
best-case |
worst-case |
average-case |
|
---|---|---|---|
search |
\(O(1)\) |
\(O(n)\) |
\(O(log\ n)\) |
insert |
\(O(1)\) |
\(O(n)\) |
\(O(log\ n)\) |
remove |
\(O(log\ n)\) |
\(O(n)\) |
\(O(log\ n)\) |
Average-case analysis#
If \(n\) distinct keys are inserted into a BST in random order, expected number of compares for basic operations is
proof: 1-1 correspondence with quick-sort
Collections / Dictionaries#
What? |
Sequential (unordered) |
Sequential (ordered) |
BST |
|
---|---|---|---|---|
search |
search for a key |
\(O(n)\) |
\(O(log\ n)\) |
\(O(h)\) |
insert |
insert a key |
\(O(n)\) |
\(O(n)\) |
\(O(h)\) |
delete |
delete a key |
\(O(n)\) |
\(O(n)\) |
\(O(h)\) |
min/max |
smallest/largest key |
\(O(n)\) |
\(O(1)\) |
\(O(h)\) |
floor/ceiling |
predecessor / successor |
\(O(n)\) |
\(O(log\ n)\) |
\(O(h)\) |
rank |
# of keys less than key |
\(O(n)\) |
\(O(log\ n)\) |
\(O(h)\)** |
(**) requires the use of ‘size’ at every node