2020-06-20 00:04:56 +08:00
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* \brief A simple tree implementation using nodes
|
|
|
|
*
|
|
|
|
* \todo update code to use C++ STL library features and OO structure
|
|
|
|
* \warning This program is a poor implementation and does not utilize any of
|
|
|
|
* the C++ STL features.
|
|
|
|
*/
|
2023-02-04 08:40:26 +08:00
|
|
|
#include <algorithm> /// for std::max
|
|
|
|
#include <iostream> /// for std::cout
|
|
|
|
#include <queue> /// for std::queue
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
using node = struct node {
|
2020-06-20 00:04:56 +08:00
|
|
|
int data;
|
|
|
|
int height;
|
|
|
|
struct node *left;
|
|
|
|
struct node *right;
|
2023-02-04 08:40:26 +08:00
|
|
|
};
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @brief creates and returns a new node
|
|
|
|
* @param[in] data value stored in the node
|
|
|
|
* @return newly created node
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
node *createNode(int data) {
|
|
|
|
node *nn = new node();
|
|
|
|
nn->data = data;
|
|
|
|
nn->height = 0;
|
2023-02-04 08:40:26 +08:00
|
|
|
nn->left = nullptr;
|
|
|
|
nn->right = nullptr;
|
2020-06-20 00:04:56 +08:00
|
|
|
return nn;
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @param[in] root the root of the tree
|
|
|
|
* @return height of tree
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
int height(node *root) {
|
2023-02-04 08:40:26 +08:00
|
|
|
if (root == nullptr) {
|
2020-06-20 00:04:56 +08:00
|
|
|
return 0;
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
2020-06-20 00:04:56 +08:00
|
|
|
return 1 + std::max(height(root->left), height(root->right));
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @param[in] root of the tree
|
|
|
|
* @return difference between height of left and right subtree
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
int getBalance(node *root) { return height(root->left) - height(root->right); }
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @param root of the tree to be rotated
|
|
|
|
* @return node after right rotation
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
node *rightRotate(node *root) {
|
|
|
|
node *t = root->left;
|
|
|
|
node *u = t->right;
|
|
|
|
t->right = root;
|
|
|
|
root->left = u;
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @param root of the tree to be rotated
|
|
|
|
* @return node after left rotation
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
node *leftRotate(node *root) {
|
|
|
|
node *t = root->right;
|
|
|
|
node *u = t->left;
|
|
|
|
t->left = root;
|
|
|
|
root->right = u;
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @param root of the tree
|
|
|
|
* @returns node with minimum value in the tree
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
node *minValue(node *root) {
|
2023-02-04 08:40:26 +08:00
|
|
|
if (root->left == nullptr) {
|
2020-06-20 00:04:56 +08:00
|
|
|
return root;
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
2020-06-20 00:04:56 +08:00
|
|
|
return minValue(root->left);
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @brief inserts a new element into AVL tree
|
|
|
|
* @param root of the tree
|
|
|
|
* @param[in] item the element to be insterted into the tree
|
|
|
|
* @return root of the updated tree
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
node *insert(node *root, int item) {
|
2023-02-04 08:40:26 +08:00
|
|
|
if (root == nullptr) {
|
|
|
|
return createNode(item);
|
|
|
|
}
|
|
|
|
if (item < root->data) {
|
2020-06-20 00:04:56 +08:00
|
|
|
root->left = insert(root->left, item);
|
2023-02-04 08:40:26 +08:00
|
|
|
} else {
|
2020-06-20 00:04:56 +08:00
|
|
|
root->right = insert(root->right, item);
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
2020-06-20 00:04:56 +08:00
|
|
|
int b = getBalance(root);
|
|
|
|
if (b > 1) {
|
2023-02-04 08:40:26 +08:00
|
|
|
if (getBalance(root->left) < 0) {
|
2020-06-20 00:04:56 +08:00
|
|
|
root->left = leftRotate(root->left); // Left-Right Case
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
|
|
|
return rightRotate(root); // Left-Left Case
|
2020-06-20 00:04:56 +08:00
|
|
|
} else if (b < -1) {
|
2023-02-04 08:40:26 +08:00
|
|
|
if (getBalance(root->right) > 0) {
|
2020-06-20 00:04:56 +08:00
|
|
|
root->right = rightRotate(root->right); // Right-Left Case
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
|
|
|
return leftRotate(root); // Right-Right Case
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @brief removes a given element from AVL tree
|
|
|
|
* @param root of the tree
|
|
|
|
* @param[in] element the element to be deleted from the tree
|
|
|
|
* @return root of the updated tree
|
|
|
|
*/
|
|
|
|
node *deleteNode(node *root, int element) {
|
|
|
|
if (root == nullptr) {
|
2020-06-20 00:04:56 +08:00
|
|
|
return root;
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
|
|
|
if (element < root->data) {
|
|
|
|
root->left = deleteNode(root->left, element);
|
|
|
|
} else if (element > root->data) {
|
|
|
|
root->right = deleteNode(root->right, element);
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
} else {
|
2020-06-20 00:04:56 +08:00
|
|
|
// Node to be deleted is leaf node or have only one Child
|
2023-02-04 08:40:26 +08:00
|
|
|
if (!root->right || !root->left) {
|
|
|
|
node *temp = !root->right ? root->left : root->right;
|
|
|
|
delete root;
|
2020-06-20 00:04:56 +08:00
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
// Node to be deleted have both left and right subtrees
|
|
|
|
node *temp = minValue(root->right);
|
|
|
|
root->data = temp->data;
|
|
|
|
root->right = deleteNode(root->right, temp->data);
|
|
|
|
}
|
|
|
|
// Balancing Tree after deletion
|
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @brief calls delete on every node
|
|
|
|
* @param root of the tree
|
|
|
|
*/
|
|
|
|
void deleteAllNodes(const node *const root) {
|
|
|
|
if (root) {
|
|
|
|
deleteAllNodes(root->left);
|
|
|
|
deleteAllNodes(root->right);
|
|
|
|
delete root;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief prints given tree in the LevelOrder
|
|
|
|
* @param[in] root of the tree
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
void levelOrder(node *root) {
|
|
|
|
std::queue<node *> q;
|
|
|
|
q.push(root);
|
|
|
|
while (!q.empty()) {
|
|
|
|
root = q.front();
|
|
|
|
std::cout << root->data << " ";
|
|
|
|
q.pop();
|
2023-02-04 08:40:26 +08:00
|
|
|
if (root->left) {
|
2020-06-20 00:04:56 +08:00
|
|
|
q.push(root->left);
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
|
|
|
if (root->right) {
|
2020-06-20 00:04:56 +08:00
|
|
|
q.push(root->right);
|
2023-02-04 08:40:26 +08:00
|
|
|
}
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-04 08:40:26 +08:00
|
|
|
/**
|
|
|
|
* @brief Main function
|
|
|
|
* @returns 0 on exit
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
int main() {
|
|
|
|
// Testing AVL Tree
|
2023-02-04 08:40:26 +08:00
|
|
|
node *root = nullptr;
|
|
|
|
int i = 0;
|
2020-06-20 00:04:56 +08:00
|
|
|
for (i = 1; i <= 7; i++) root = insert(root, i);
|
|
|
|
std::cout << "LevelOrder: ";
|
|
|
|
levelOrder(root);
|
|
|
|
root = deleteNode(root, 1); // Deleting key with value 1
|
|
|
|
std::cout << "\nLevelOrder: ";
|
|
|
|
levelOrder(root);
|
|
|
|
root = deleteNode(root, 4); // Deletin key with value 4
|
|
|
|
std::cout << "\nLevelOrder: ";
|
|
|
|
levelOrder(root);
|
2023-02-04 08:40:26 +08:00
|
|
|
deleteAllNodes(root);
|
2020-06-20 00:04:56 +08:00
|
|
|
return 0;
|
|
|
|
}
|