working on assn06

This commit is contained in:
Rushil Umaretiya 2023-11-17 19:28:35 -05:00
parent 79203cf35b
commit 883a78b824
No known key found for this signature in database
GPG Key ID: 4E8FAF9C926AF959
4 changed files with 412 additions and 0 deletions

287
src/assn06/AVLTree.java Normal file
View File

@ -0,0 +1,287 @@
package assn06;
public class AVLTree<T extends Comparable<T>> implements SelfBalancingBST<T> {
// Fields
private T _value;
private AVLTree<T> _left;
private AVLTree<T> _right;
private int _height;
private int _size;
public AVLTree() {
_value = null;
_left = null;
_right = null;
_height = -1;
_size = 0;
}
/**
* Rotates the tree left and returns
* AVLTree root for rotated result.
*/
private AVLTree<T> rotateLeft() {
if (isEmpty() || _right == null) {
return this;
}
AVLTree<T> newRoot = _right;
AVLTree<T> temp = newRoot._left;
newRoot._left = this;
_right = temp;
updateHeight();
newRoot.updateHeight();
return newRoot;
}
/**
* Rotates the tree right and returns
* AVLTree root for rotated result.
*/
private AVLTree<T> rotateRight() {
if (isEmpty() || _left == null) {
return this;
}
AVLTree<T> newRoot = _left;
AVLTree<T> temp = newRoot._right;
newRoot._right = this;
_left = temp;
updateHeight();
newRoot.updateHeight();
return newRoot;
}
@Override
public boolean isEmpty() {
return size() == 0;
}
@Override
public int height() {
return _height;
}
@Override
public int size() {
return _size;
}
@Override
public SelfBalancingBST<T> insert(T element) {
if (isEmpty()) {
_value = element;
_left = new AVLTree<>();
_right = new AVLTree<>();
_height = 0;
_size = 1;
return this;
}
int compareResult = element.compareTo(_value);
if (compareResult < 0) {
_left = (AVLTree<T>) _left.insert(element);
} else if (compareResult > 0) {
_right = (AVLTree<T>) _right.insert(element);
} else {
// Duplicate values are not allowed in this AVL tree implementation
return this;
}
updateHeight();
_size++;
int balance = getBalance();
if (balance > 1) {
if (_left.getBalance() < 0) {
_left = _left.rotateLeft();
}
return rotateRight();
}
else if (balance < -1) {
if (_right.getBalance() > 0) {
_right = _right.rotateRight();
}
return rotateLeft();
}
return this;
}
@Override
public SelfBalancingBST<T> remove(T element) {
if (isEmpty()) {
return this;
}
int compareResult = element.compareTo(_value);
if (compareResult < 0) {
_left = (AVLTree<T>) _left.remove(element);
} else if (compareResult > 0) {
_right = (AVLTree<T>) _right.remove(element);
} else {
// Node with only one child or no child
if (_left.isEmpty() || _right.isEmpty()) {
AVLTree<T> temp = _left.isEmpty() ? _right : _left;
if (temp.isEmpty()) {
return new AVLTree<>();
} else {
return temp;
}
} else {
_value = _right.findMin();
_right = (AVLTree<T>) _right.remove(_value);
}
}
updateHeight();
_size--;
int balance = getBalance();
if (balance > 1) {
if (_left.getBalance() < 0) {
_left = _left.rotateLeft();
}
return rotateRight();
}
else if (balance < -1) {
if (_right.getBalance() > 0) {
_right = _right.rotateRight();
}
return rotateLeft();
}
return this;
}
@Override
public T findMin() {
if (isEmpty()) {
throw new RuntimeException("Illegal operation on empty tree");
}
if (_left == null || _left.isEmpty()) {
return _value;
} else {
return _left.findMin();
}
}
@Override
public T findMax() {
if (isEmpty()) {
throw new RuntimeException("Illegal operation on empty tree");
}
// Check if _right is not null before calling isEmpty on it
if (_right == null || _right.isEmpty()) {
return _value;
} else {
return _right.findMax();
}
}
@Override
public boolean contains(T element) {
if (isEmpty()) {
return false;
}
if (_value.compareTo(element) > 0) {
return _left.contains(element);
} else if (_value.compareTo(element) < 0) {
return _right.contains(element);
} else {
return true;
}
}
@Override
public T getValue() {
return _value;
}
@Override
public SelfBalancingBST<T> getLeft() {
if (isEmpty()) {
return null;
}
return _left;
}
@Override
public SelfBalancingBST<T> getRight() {
if (isEmpty()) {
return null;
}
return _right;
}
private void updateHeight() {
int leftHeight = (_left != null) ? _left.height() : -1;
int rightHeight = (_right != null) ? _right.height() : -1;
_height = 1 + Math.max(leftHeight, rightHeight);
}
private int getBalance() {
if (isEmpty()) {
return 0;
}
int leftHeight = _left != null ? _left.height() : -1;
int rightHeight = _right != null ? _right.height() : -1;
return leftHeight - rightHeight;
}
public String toString() {
if (isEmpty()) {
return "";
}
String result = "";
if (_left != null) {
result += _left.toString();
}
result += _value + " ";
if (_right != null) {
result += _right.toString();
}
return result;
}
/**
* Checks if the tree is balanced.
*
* @return true if the tree is balanced, false otherwise.
*/
public boolean isBalanced() {
if (isEmpty()) {
return true;
}
int balance = getBalance();
if (Math.abs(balance) > 1) {
return false;
} else {
boolean leftBalanced = _left == null || _left.isBalanced();
boolean rightBalanced = _right == null || _right.isBalanced();
return leftBalanced && rightBalanced;
}
}
}

Binary file not shown.

44
src/assn06/Main.java Normal file
View File

@ -0,0 +1,44 @@
package assn06;
public class Main {
public static void main(String[] args) {
// Create a new empty tree.
SelfBalancingBST<Integer> avl_bst = new AVLTree<Integer>();
// Insert 50 random integers.
// Note how we need to capture the result of insert back into
// the variable avl_bst because the post-insertion root that is
// returned may be different from the original root because of the insertion.
// result should be about 6.
for (int i = 0; i < 50; i++) {
avl_bst = avl_bst.insert((int) (Math.random() * 100));
}
System.out.println(avl_bst.height());
// Now insert 50 integers in increasing order which would
// cause a simple BST to become very tall but for our
// self-balancing tree won't be too bad (should be 7)
for (int i = 0; i < 50; i++) {
avl_bst = avl_bst.insert(i);
}
System.out.println(avl_bst.height());
// System.out.println(avl_bst.toString());
System.out.println("test cases:");
SelfBalancingBST<Integer> avl = new AVLTree<Integer>();
int[] numbers = { 20, 11, 50, 4, 6, 15, 3, 16, 17 };
for (int i = 0; i < numbers.length; i++) {
avl = avl.insert(numbers[i]);
}
System.out.println(avl.height());
System.out.println(avl.toString());
System.out.println(avl.isBalanced());
}
}

View File

@ -0,0 +1,81 @@
package assn06;
public interface SelfBalancingBST<T extends Comparable<T>> {
/**
* @return true if the tree is empty
*/
boolean isEmpty();
/**
* @return height of the tree.
*/
int height();
/**
* @return the number of elements in the tree
*/
int size();
/**
* Inserts element into tree and returns resulting
* tree (i.e. root) after insertion. Depending on implementation,
* this may or may not be the same object that you started with.
* @param element to be added to the tree
* @return resulting tree after insertion
**/
SelfBalancingBST<T> insert(T element);
/**
* Removes element from tree and returns resulting
* tree after removal. Depending on implementation,
* this may or may not be the same object that you
* started with. If element is not in the tree, the
* tree should remain unchanged and return itself.
* @param element to be removed from the tree
* @return resulting tree after removal
**/
SelfBalancingBST<T> remove(T element);
/**
* Throws a RuntimeException if called on an empty tree.
* @return the smallest value in the tree
*/
T findMin();
/**
* Throws a RuntimeException if called on an empty tree.
* @return the smallest value in the tree.
*/
T findMax();
/**
* @param element whose presence in this tree is to be tested
* @return true if this tree contains the specified element
*/
boolean contains(T element);
/**
* @return value at the top of the tree.
* If a tree is empty, its value is null.
*/
T getValue();
/**
* @return the left child of the tree.
* Throws a RuntimeException if the tree is empty.
*/
SelfBalancingBST<T> getLeft();
/**
* @return the right child of the tree.
* Throws a RuntimeException if the tree is empty.
*/
SelfBalancingBST<T> getRight();
/**
* @return true if the tree is balanced.
*/
boolean isBalanced();
}