looking at the avltree implementation; ok, I think I understand it enough to edit it.
This commit is contained in:
parent
3653b24476
commit
78d875aac1
|
@ -1,7 +1,12 @@
|
||||||
// ----------------------------------------------------------------------
|
// ----------------------------------------------------------------------
|
||||||
// dyoo: the following code comes from the Google Closure Library. I've done
|
// dyoo: the following code comes from the Google Closure Library. I've done
|
||||||
// light edits to flatten the namespace from goog.structs to just
|
// edits to flatten the namespace from goog.structs to just
|
||||||
// AvlTree.
|
// AvlTree, commented out inorderTraverse and reverseOrderTraverse.
|
||||||
|
//
|
||||||
|
// I'm considering changing the code to work with CPSed control flow.
|
||||||
|
//
|
||||||
|
// ----------------------------------------------------------------------
|
||||||
|
// Original license follows:
|
||||||
// ----------------------------------------------------------------------
|
// ----------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
@ -58,8 +63,8 @@
|
||||||
* @implements {Collection}
|
* @implements {Collection}
|
||||||
*/
|
*/
|
||||||
var AvlTree = function(opt_comparator) {
|
var AvlTree = function(opt_comparator) {
|
||||||
this.comparator_ = opt_comparator ||
|
this.comparator_ = opt_comparator ||
|
||||||
AvlTree.DEFAULT_COMPARATOR_;
|
AvlTree.DEFAULT_COMPARATOR_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -73,12 +78,12 @@ var AvlTree = function(opt_comparator) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.DEFAULT_COMPARATOR_ = function(a, b) {
|
AvlTree.DEFAULT_COMPARATOR_ = function(a, b) {
|
||||||
if (String(a) < String(b)) {
|
if (String(a) < String(b)) {
|
||||||
return -1;
|
return -1;
|
||||||
} else if (String(a) > String(b)) {
|
} else if (String(a) > String(b)) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -142,54 +147,54 @@ AvlTree.prototype.count_ = 0;
|
||||||
* @return {boolean} Whether value was inserted into the tree.
|
* @return {boolean} Whether value was inserted into the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.add = function(value) {
|
AvlTree.prototype.add = function(value) {
|
||||||
// If the tree is empty, create a root node with the specified value
|
// If the tree is empty, create a root node with the specified value
|
||||||
if (this.root_ == null) {
|
if (this.root_ == null) {
|
||||||
this.root_ = new AvlTree.Node(value);
|
this.root_ = new AvlTree.Node(value);
|
||||||
this.minNode_ = this.root_;
|
this.minNode_ = this.root_;
|
||||||
this.maxNode_ = this.root_;
|
this.maxNode_ = this.root_;
|
||||||
this.count_ = 1;
|
this.count_ = 1;
|
||||||
return true;
|
return true;
|
||||||
}
|
|
||||||
|
|
||||||
// Assume a node is not added and change status when one is
|
|
||||||
var retStatus = false;
|
|
||||||
|
|
||||||
// Depth traverse the tree and insert the value if we reach a null node
|
|
||||||
this.traverse_(function(node) {
|
|
||||||
var retNode = null;
|
|
||||||
if (this.comparator_(node.value, value) > 0) {
|
|
||||||
retNode = node.left;
|
|
||||||
if (node.left == null) {
|
|
||||||
var newNode = new AvlTree.Node(value, node);
|
|
||||||
node.left = newNode;
|
|
||||||
if (node == this.minNode_) {
|
|
||||||
this.minNode_ = newNode;
|
|
||||||
}
|
|
||||||
retStatus = true; // Value was added to tree
|
|
||||||
this.balance_(node); // Maintain the AVL-tree balance
|
|
||||||
}
|
|
||||||
} else if (this.comparator_(node.value, value) < 0) {
|
|
||||||
retNode = node.right;
|
|
||||||
if (node.right == null) {
|
|
||||||
var newNode = new AvlTree.Node(value, node);
|
|
||||||
node.right = newNode;
|
|
||||||
if (node == this.maxNode_) {
|
|
||||||
this.maxNode_ = newNode;
|
|
||||||
}
|
|
||||||
retStatus = true; // Value was added to tree
|
|
||||||
this.balance_(node); // Maintain the AVL-tree balance
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
});
|
|
||||||
|
|
||||||
// If a node was added, increment count
|
// Assume a node is not added and change status when one is
|
||||||
if (retStatus) {
|
var retStatus = false;
|
||||||
this.count_ += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return true if a node was added, false otherwise
|
// Depth traverse the tree and insert the value if we reach a null node
|
||||||
return retStatus;
|
this.traverse_(function(node) {
|
||||||
|
var retNode = null;
|
||||||
|
if (this.comparator_(node.value, value) > 0) {
|
||||||
|
retNode = node.left;
|
||||||
|
if (node.left == null) {
|
||||||
|
var newNode = new AvlTree.Node(value, node);
|
||||||
|
node.left = newNode;
|
||||||
|
if (node == this.minNode_) {
|
||||||
|
this.minNode_ = newNode;
|
||||||
|
}
|
||||||
|
retStatus = true; // Value was added to tree
|
||||||
|
this.balance_(node); // Maintain the AVL-tree balance
|
||||||
|
}
|
||||||
|
} else if (this.comparator_(node.value, value) < 0) {
|
||||||
|
retNode = node.right;
|
||||||
|
if (node.right == null) {
|
||||||
|
var newNode = new AvlTree.Node(value, node);
|
||||||
|
node.right = newNode;
|
||||||
|
if (node == this.maxNode_) {
|
||||||
|
this.maxNode_ = newNode;
|
||||||
|
}
|
||||||
|
retStatus = true; // Value was added to tree
|
||||||
|
this.balance_(node); // Maintain the AVL-tree balance
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
|
});
|
||||||
|
|
||||||
|
// If a node was added, increment count
|
||||||
|
if (retStatus) {
|
||||||
|
this.count_ += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return true if a node was added, false otherwise
|
||||||
|
return retStatus;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -203,31 +208,31 @@ AvlTree.prototype.add = function(value) {
|
||||||
* the tree.
|
* the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.remove = function(value) {
|
AvlTree.prototype.remove = function(value) {
|
||||||
// Assume the value is not removed and set the value when it is removed
|
// Assume the value is not removed and set the value when it is removed
|
||||||
var retValue = null;
|
var retValue = null;
|
||||||
|
|
||||||
// Depth traverse the tree and remove the value if we find it
|
// Depth traverse the tree and remove the value if we find it
|
||||||
this.traverse_(function(node) {
|
this.traverse_(function(node) {
|
||||||
var retNode = null;
|
var retNode = null;
|
||||||
if (this.comparator_(node.value, value) > 0) {
|
if (this.comparator_(node.value, value) > 0) {
|
||||||
retNode = node.left;
|
retNode = node.left;
|
||||||
} else if (this.comparator_(node.value, value) < 0) {
|
} else if (this.comparator_(node.value, value) < 0) {
|
||||||
retNode = node.right;
|
retNode = node.right;
|
||||||
} else {
|
} else {
|
||||||
retValue = node.value;
|
retValue = node.value;
|
||||||
this.removeNode_(node);
|
this.removeNode_(node);
|
||||||
|
}
|
||||||
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
|
});
|
||||||
|
|
||||||
|
// If a node was removed, decrement count.
|
||||||
|
if (retValue) {
|
||||||
|
// Had traverse_() cleared the tree, set to 0.
|
||||||
|
this.count_ = this.root_ ? this.count_ - 1 : 0;
|
||||||
}
|
}
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
});
|
|
||||||
|
|
||||||
// If a node was removed, decrement count.
|
// Return the value that was removed, null if the value was not in the tree
|
||||||
if (retValue) {
|
return retValue;
|
||||||
// Had traverse_() cleared the tree, set to 0.
|
|
||||||
this.count_ = this.root_ ? this.count_ - 1 : 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the value that was removed, null if the value was not in the tree
|
|
||||||
return retValue;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -235,10 +240,10 @@ AvlTree.prototype.remove = function(value) {
|
||||||
* Removes all nodes from the tree.
|
* Removes all nodes from the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.clear = function() {
|
AvlTree.prototype.clear = function() {
|
||||||
this.root_ = null;
|
this.root_ = null;
|
||||||
this.minNode_ = null;
|
this.minNode_ = null;
|
||||||
this.maxNode_ = null;
|
this.maxNode_ = null;
|
||||||
this.count_ = 0;
|
this.count_ = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -250,24 +255,24 @@ AvlTree.prototype.clear = function() {
|
||||||
* @return {boolean} Whether the tree contains a node with the specified value.
|
* @return {boolean} Whether the tree contains a node with the specified value.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.contains = function(value) {
|
AvlTree.prototype.contains = function(value) {
|
||||||
// Assume the value is not in the tree and set this value if it is found
|
// Assume the value is not in the tree and set this value if it is found
|
||||||
var isContained = false;
|
var isContained = false;
|
||||||
|
|
||||||
// Depth traverse the tree and set isContained if we find the node
|
// Depth traverse the tree and set isContained if we find the node
|
||||||
this.traverse_(function(node) {
|
this.traverse_(function(node) {
|
||||||
var retNode = null;
|
var retNode = null;
|
||||||
if (this.comparator_(node.value, value) > 0) {
|
if (this.comparator_(node.value, value) > 0) {
|
||||||
retNode = node.left;
|
retNode = node.left;
|
||||||
} else if (this.comparator_(node.value, value) < 0) {
|
} else if (this.comparator_(node.value, value) < 0) {
|
||||||
retNode = node.right;
|
retNode = node.right;
|
||||||
} else {
|
} else {
|
||||||
isContained = true;
|
isContained = true;
|
||||||
}
|
}
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
});
|
});
|
||||||
|
|
||||||
// Return true if the value is contained in the tree, false otherwise
|
// Return true if the value is contained in the tree, false otherwise
|
||||||
return isContained;
|
return isContained;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -277,7 +282,7 @@ AvlTree.prototype.contains = function(value) {
|
||||||
* @return {number} The number of values stored in the tree.
|
* @return {number} The number of values stored in the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getCount = function() {
|
AvlTree.prototype.getCount = function() {
|
||||||
return this.count_;
|
return this.count_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -288,7 +293,7 @@ AvlTree.prototype.getCount = function() {
|
||||||
* @return {*} The minimum value contained in the tree.
|
* @return {*} The minimum value contained in the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getMinimum = function() {
|
AvlTree.prototype.getMinimum = function() {
|
||||||
return this.getMinNode_().value;
|
return this.getMinNode_().value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -299,7 +304,7 @@ AvlTree.prototype.getMinimum = function() {
|
||||||
* @return {*} The maximum value contained in the tree.
|
* @return {*} The maximum value contained in the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getMaximum = function() {
|
AvlTree.prototype.getMaximum = function() {
|
||||||
return this.getMaxNode_().value;
|
return this.getMaxNode_().value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -311,7 +316,7 @@ AvlTree.prototype.getMaximum = function() {
|
||||||
* @return {number} The height of the tree.
|
* @return {number} The height of the tree.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getHeight = function() {
|
AvlTree.prototype.getHeight = function() {
|
||||||
return this.root_ ? this.root_.height : 0;
|
return this.root_ ? this.root_.height : 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -321,11 +326,11 @@ AvlTree.prototype.getHeight = function() {
|
||||||
* @return {Array} An array containing all of the trees values in sorted order.
|
* @return {Array} An array containing all of the trees values in sorted order.
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getValues = function() {
|
AvlTree.prototype.getValues = function() {
|
||||||
var ret = [];
|
var ret = [];
|
||||||
this.inOrderTraverse(function(value) {
|
this.inOrderTraverse(function(value) {
|
||||||
ret.push(value);
|
ret.push(value);
|
||||||
});
|
});
|
||||||
return ret;
|
return ret;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -341,106 +346,106 @@ AvlTree.prototype.getValues = function() {
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.inOrderTraverse =
|
AvlTree.prototype.inOrderTraverse =
|
||||||
function(func, opt_startValue) {
|
function(func, opt_startValue) {
|
||||||
// If our tree is empty, return immediately
|
// If our tree is empty, return immediately
|
||||||
if (!this.root_) {
|
if (!this.root_) {
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
// Depth traverse the tree to find node to begin in-order traversal from
|
|
||||||
var startNode;
|
|
||||||
if (opt_startValue) {
|
|
||||||
this.traverse_(function(node) {
|
|
||||||
var retNode = null;
|
|
||||||
if (this.comparator_(node.value, opt_startValue) > 0) {
|
|
||||||
retNode = node.left;
|
|
||||||
startNode = node;
|
|
||||||
} else if (this.comparator_(node.value, opt_startValue) < 0) {
|
|
||||||
retNode = node.right;
|
|
||||||
} else {
|
|
||||||
startNode = node;
|
|
||||||
}
|
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
startNode = this.getMinNode_();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Traverse the tree and call func on each traversed node's value
|
|
||||||
var node = startNode, prev = startNode.left ? startNode.left : startNode;
|
|
||||||
while (node != null) {
|
|
||||||
if (node.left != null && node.left != prev && node.right != prev) {
|
|
||||||
node = node.left;
|
|
||||||
} else {
|
|
||||||
if (node.right != prev) {
|
|
||||||
if (func(node.value)) {
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
var temp = node;
|
|
||||||
node = node.right != null && node.right != prev ?
|
|
||||||
node.right :
|
|
||||||
node.parent;
|
|
||||||
prev = temp;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
// Depth traverse the tree to find node to begin in-order traversal from
|
||||||
/**
|
var startNode;
|
||||||
* Performs a reverse-order traversal of the tree and calls {@code func} with
|
if (opt_startValue) {
|
||||||
* each traversed node, optionally starting from the largest node with a value
|
this.traverse_(function(node) {
|
||||||
* <= to the specified start value. The traversal ends after traversing the
|
var retNode = null;
|
||||||
* tree's minimum node or when func returns a value that evaluates to true.
|
if (this.comparator_(node.value, opt_startValue) > 0) {
|
||||||
*
|
retNode = node.left;
|
||||||
* @param {Function} func Function to call on each traversed node.
|
startNode = node;
|
||||||
* @param {Object=} opt_startValue If specified, traversal will begin on the
|
} else if (this.comparator_(node.value, opt_startValue) < 0) {
|
||||||
* node with the largest value <= opt_startValue.
|
retNode = node.right;
|
||||||
*/
|
} else {
|
||||||
AvlTree.prototype.reverseOrderTraverse =
|
startNode = node;
|
||||||
function(func, opt_startValue) {
|
}
|
||||||
// If our tree is empty, return immediately
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
if (!this.root_) {
|
});
|
||||||
return;
|
} else {
|
||||||
}
|
startNode = this.getMinNode_();
|
||||||
|
|
||||||
// Depth traverse the tree to find node to begin reverse-order traversal from
|
|
||||||
var startNode;
|
|
||||||
if (opt_startValue) {
|
|
||||||
this.traverse_(goog.bind(function(node) {
|
|
||||||
var retNode = null;
|
|
||||||
if (this.comparator_(node.value, opt_startValue) > 0) {
|
|
||||||
retNode = node.left;
|
|
||||||
} else if (this.comparator_(node.value, opt_startValue) < 0) {
|
|
||||||
retNode = node.right;
|
|
||||||
startNode = node;
|
|
||||||
} else {
|
|
||||||
startNode = node;
|
|
||||||
}
|
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
}, this));
|
|
||||||
} else {
|
|
||||||
startNode = this.getMaxNode_();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Traverse the tree and call func on each traversed node's value
|
|
||||||
var node = startNode, prev = startNode.right ? startNode.right : startNode;
|
|
||||||
while (node != null) {
|
|
||||||
if (node.right != null && node.right != prev && node.left != prev) {
|
|
||||||
node = node.right;
|
|
||||||
} else {
|
|
||||||
if (node.left != prev) {
|
|
||||||
if (func(node.value)) {
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
var temp = node;
|
// Traverse the tree and call func on each traversed node's value
|
||||||
node = node.left != null && node.left != prev ?
|
var node = startNode, prev = startNode.left ? startNode.left : startNode;
|
||||||
node.left :
|
while (node != null) {
|
||||||
node.parent;
|
if (node.left != null && node.left != prev && node.right != prev) {
|
||||||
prev = temp;
|
node = node.left;
|
||||||
}
|
} else {
|
||||||
}
|
if (node.right != prev) {
|
||||||
};
|
if (func(node.value)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var temp = node;
|
||||||
|
node = node.right != null && node.right != prev ?
|
||||||
|
node.right :
|
||||||
|
node.parent;
|
||||||
|
prev = temp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// /**
|
||||||
|
// * Performs a reverse-order traversal of the tree and calls {@code func} with
|
||||||
|
// * each traversed node, optionally starting from the largest node with a value
|
||||||
|
// * <= to the specified start value. The traversal ends after traversing the
|
||||||
|
// * tree's minimum node or when func returns a value that evaluates to true.
|
||||||
|
// *
|
||||||
|
// * @param {Function} func Function to call on each traversed node.
|
||||||
|
// * @param {Object=} opt_startValue If specified, traversal will begin on the
|
||||||
|
// * node with the largest value <= opt_startValue.
|
||||||
|
// */
|
||||||
|
// AvlTree.prototype.reverseOrderTraverse =
|
||||||
|
// function(func, opt_startValue) {
|
||||||
|
// // If our tree is empty, return immediately
|
||||||
|
// if (!this.root_) {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// // Depth traverse the tree to find node to begin reverse-order traversal from
|
||||||
|
// var startNode;
|
||||||
|
// if (opt_startValue) {
|
||||||
|
// this.traverse_(goog.bind(function(node) {
|
||||||
|
// var retNode = null;
|
||||||
|
// if (this.comparator_(node.value, opt_startValue) > 0) {
|
||||||
|
// retNode = node.left;
|
||||||
|
// } else if (this.comparator_(node.value, opt_startValue) < 0) {
|
||||||
|
// retNode = node.right;
|
||||||
|
// startNode = node;
|
||||||
|
// } else {
|
||||||
|
// startNode = node;
|
||||||
|
// }
|
||||||
|
// return retNode; // If null, we'll stop traversing the tree
|
||||||
|
// }, this));
|
||||||
|
// } else {
|
||||||
|
// startNode = this.getMaxNode_();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// // Traverse the tree and call func on each traversed node's value
|
||||||
|
// var node = startNode, prev = startNode.right ? startNode.right : startNode;
|
||||||
|
// while (node != null) {
|
||||||
|
// if (node.right != null && node.right != prev && node.left != prev) {
|
||||||
|
// node = node.right;
|
||||||
|
// } else {
|
||||||
|
// if (node.left != prev) {
|
||||||
|
// if (func(node.value)) {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// var temp = node;
|
||||||
|
// node = node.left != null && node.left != prev ?
|
||||||
|
// node.left :
|
||||||
|
// node.parent;
|
||||||
|
// prev = temp;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// };
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -461,12 +466,29 @@ AvlTree.prototype.reverseOrderTraverse =
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.traverse_ =
|
AvlTree.prototype.traverse_ =
|
||||||
function(traversalFunc, opt_startNode, opt_endNode) {
|
function(traversalFunc, opt_startNode, opt_endNode) {
|
||||||
var node = opt_startNode ? opt_startNode : this.root_;
|
var node = opt_startNode ? opt_startNode : this.root_;
|
||||||
var endNode = opt_endNode ? opt_endNode : null;
|
var endNode = opt_endNode ? opt_endNode : null;
|
||||||
while (node && node != endNode) {
|
while (node && node != endNode) {
|
||||||
node = traversalFunc.call(this, node);
|
node = traversalFunc.call(this, node);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// CPS'ed version of the traverse function
|
||||||
|
AvlTree.prototype.traverse_k_ =
|
||||||
|
function(traversalFunc_k, opt_startNode, opt_endNode, k) {
|
||||||
|
var node = opt_startNode ? opt_startNode : this.root_;
|
||||||
|
var endNode = opt_endNode ? opt_endNode : null;
|
||||||
|
var loop = function(node) {
|
||||||
|
if (node && node != endNode) {
|
||||||
|
traversalFunc_k.call(this, node, loop);
|
||||||
|
} else {
|
||||||
|
k();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return loop(node);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -481,36 +503,36 @@ AvlTree.prototype.traverse_ =
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.balance_ = function(node) {
|
AvlTree.prototype.balance_ = function(node) {
|
||||||
|
|
||||||
this.traverse_(function(node) {
|
this.traverse_(function(node) {
|
||||||
// Calculate the left and right node's heights
|
// Calculate the left and right node's heights
|
||||||
var lh = node.left ? node.left.height : 0;
|
var lh = node.left ? node.left.height : 0;
|
||||||
var rh = node.right ? node.right.height : 0;
|
var rh = node.right ? node.right.height : 0;
|
||||||
|
|
||||||
// Rotate tree rooted at this node if it is not AVL-tree balanced
|
// Rotate tree rooted at this node if it is not AVL-tree balanced
|
||||||
if (lh - rh > 1) {
|
if (lh - rh > 1) {
|
||||||
if (node.left.right && (!node.left.left ||
|
if (node.left.right && (!node.left.left ||
|
||||||
node.left.left.height < node.left.right.height)) {
|
node.left.left.height < node.left.right.height)) {
|
||||||
this.leftRotate_(node.left);
|
this.leftRotate_(node.left);
|
||||||
}
|
}
|
||||||
this.rightRotate_(node);
|
this.rightRotate_(node);
|
||||||
} else if (rh - lh > 1) {
|
} else if (rh - lh > 1) {
|
||||||
if (node.right.left && (!node.right.right ||
|
if (node.right.left && (!node.right.right ||
|
||||||
node.right.right.height < node.right.left.height)) {
|
node.right.right.height < node.right.left.height)) {
|
||||||
this.rightRotate_(node.right);
|
this.rightRotate_(node.right);
|
||||||
}
|
}
|
||||||
this.leftRotate_(node);
|
this.leftRotate_(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Recalculate the left and right node's heights
|
// Recalculate the left and right node's heights
|
||||||
lh = node.left ? node.left.height : 0;
|
lh = node.left ? node.left.height : 0;
|
||||||
rh = node.right ? node.right.height : 0;
|
rh = node.right ? node.right.height : 0;
|
||||||
|
|
||||||
// Set this node's height
|
// Set this node's height
|
||||||
node.height = Math.max(lh, rh) + 1;
|
node.height = Math.max(lh, rh) + 1;
|
||||||
|
|
||||||
// Traverse up tree and balance parent
|
// Traverse up tree and balance parent
|
||||||
return node.parent;
|
return node.parent;
|
||||||
}, node);
|
}, node);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -522,24 +544,24 @@ AvlTree.prototype.balance_ = function(node) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.leftRotate_ = function(node) {
|
AvlTree.prototype.leftRotate_ = function(node) {
|
||||||
// Re-assign parent-child references for the parent of the node being removed
|
// Re-assign parent-child references for the parent of the node being removed
|
||||||
if (node.isLeftChild()) {
|
if (node.isLeftChild()) {
|
||||||
node.parent.left = node.right;
|
node.parent.left = node.right;
|
||||||
node.right.parent = node.parent;
|
node.right.parent = node.parent;
|
||||||
} else if (node.isRightChild()) {
|
} else if (node.isRightChild()) {
|
||||||
node.parent.right = node.right;
|
node.parent.right = node.right;
|
||||||
node.right.parent = node.parent;
|
node.right.parent = node.parent;
|
||||||
} else {
|
} else {
|
||||||
this.root_ = node.right;
|
this.root_ = node.right;
|
||||||
this.root_.parent = null;
|
this.root_.parent = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Re-assign parent-child references for the child of the node being removed
|
// Re-assign parent-child references for the child of the node being removed
|
||||||
var temp = node.right;
|
var temp = node.right;
|
||||||
node.right = node.right.left;
|
node.right = node.right.left;
|
||||||
if (node.right != null) node.right.parent = node;
|
if (node.right != null) node.right.parent = node;
|
||||||
temp.left = node;
|
temp.left = node;
|
||||||
node.parent = temp;
|
node.parent = temp;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -550,24 +572,24 @@ AvlTree.prototype.leftRotate_ = function(node) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.rightRotate_ = function(node) {
|
AvlTree.prototype.rightRotate_ = function(node) {
|
||||||
// Re-assign parent-child references for the parent of the node being removed
|
// Re-assign parent-child references for the parent of the node being removed
|
||||||
if (node.isLeftChild()) {
|
if (node.isLeftChild()) {
|
||||||
node.parent.left = node.left;
|
node.parent.left = node.left;
|
||||||
node.left.parent = node.parent;
|
node.left.parent = node.parent;
|
||||||
} else if (node.isRightChild()) {
|
} else if (node.isRightChild()) {
|
||||||
node.parent.right = node.left;
|
node.parent.right = node.left;
|
||||||
node.left.parent = node.parent;
|
node.left.parent = node.parent;
|
||||||
} else {
|
} else {
|
||||||
this.root_ = node.left;
|
this.root_ = node.left;
|
||||||
this.root_.parent = null;
|
this.root_.parent = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Re-assign parent-child references for the child of the node being removed
|
// Re-assign parent-child references for the child of the node being removed
|
||||||
var temp = node.left;
|
var temp = node.left;
|
||||||
node.left = node.left.right;
|
node.left = node.left.right;
|
||||||
if (node.left != null) node.left.parent = node;
|
if (node.left != null) node.left.parent = node;
|
||||||
temp.right = node;
|
temp.right = node;
|
||||||
node.parent = temp;
|
node.parent = temp;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -579,65 +601,65 @@ AvlTree.prototype.rightRotate_ = function(node) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.removeNode_ = function(node) {
|
AvlTree.prototype.removeNode_ = function(node) {
|
||||||
// Perform normal binary tree node removal, but balance the tree, starting
|
// Perform normal binary tree node removal, but balance the tree, starting
|
||||||
// from where we removed the node
|
// from where we removed the node
|
||||||
if (node.left != null || node.right != null) {
|
if (node.left != null || node.right != null) {
|
||||||
var b = null; // Node to begin balance from
|
var b = null; // Node to begin balance from
|
||||||
var r; // Node to replace the node being removed
|
var r; // Node to replace the node being removed
|
||||||
if (node.left != null) {
|
if (node.left != null) {
|
||||||
r = this.getMaxNode_(node.left);
|
r = this.getMaxNode_(node.left);
|
||||||
if (r != node.left) {
|
if (r != node.left) {
|
||||||
r.parent.right = r.left;
|
r.parent.right = r.left;
|
||||||
if (r.left) r.left.parent = r.parent;
|
if (r.left) r.left.parent = r.parent;
|
||||||
r.left = node.left;
|
r.left = node.left;
|
||||||
r.left.parent = r;
|
r.left.parent = r;
|
||||||
b = r.parent;
|
b = r.parent;
|
||||||
}
|
}
|
||||||
r.parent = node.parent;
|
r.parent = node.parent;
|
||||||
r.right = node.right;
|
r.right = node.right;
|
||||||
if (r.right) r.right.parent = r;
|
if (r.right) r.right.parent = r;
|
||||||
if (node == this.maxNode_) this.maxNode_ = r;
|
if (node == this.maxNode_) this.maxNode_ = r;
|
||||||
} else {
|
} else {
|
||||||
r = this.getMinNode_(node.right);
|
r = this.getMinNode_(node.right);
|
||||||
if (r != node.right) {
|
if (r != node.right) {
|
||||||
r.parent.left = r.right;
|
r.parent.left = r.right;
|
||||||
if (r.right) r.right.parent = r.parent;
|
if (r.right) r.right.parent = r.parent;
|
||||||
r.right = node.right;
|
r.right = node.right;
|
||||||
r.right.parent = r;
|
r.right.parent = r;
|
||||||
b = r.parent;
|
b = r.parent;
|
||||||
}
|
}
|
||||||
r.parent = node.parent;
|
r.parent = node.parent;
|
||||||
r.left = node.left;
|
r.left = node.left;
|
||||||
if (r.left) r.left.parent = r;
|
if (r.left) r.left.parent = r;
|
||||||
if (node == this.minNode_) this.minNode_ = r;
|
if (node == this.minNode_) this.minNode_ = r;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the parent of the node being removed to point to its replace
|
// Update the parent of the node being removed to point to its replace
|
||||||
if (node.isLeftChild()) {
|
if (node.isLeftChild()) {
|
||||||
node.parent.left = r;
|
node.parent.left = r;
|
||||||
} else if (node.isRightChild()) {
|
} else if (node.isRightChild()) {
|
||||||
node.parent.right = r;
|
node.parent.right = r;
|
||||||
} else {
|
} else {
|
||||||
this.root_ = r;
|
this.root_ = r;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Balance the tree
|
// Balance the tree
|
||||||
this.balance_(b ? b : r);
|
this.balance_(b ? b : r);
|
||||||
} else {
|
|
||||||
// If the node is a leaf, remove it and balance starting from its parent
|
|
||||||
if (node.isLeftChild()) {
|
|
||||||
this.special = 1;
|
|
||||||
node.parent.left = null;
|
|
||||||
if (node == this.minNode_) this.minNode_ = node.parent;
|
|
||||||
this.balance_(node.parent);
|
|
||||||
} else if (node.isRightChild()) {
|
|
||||||
node.parent.right = null;
|
|
||||||
if (node == this.maxNode_) this.maxNode_ = node.parent;
|
|
||||||
this.balance_(node.parent);
|
|
||||||
} else {
|
} else {
|
||||||
this.clear();
|
// If the node is a leaf, remove it and balance starting from its parent
|
||||||
|
if (node.isLeftChild()) {
|
||||||
|
this.special = 1;
|
||||||
|
node.parent.left = null;
|
||||||
|
if (node == this.minNode_) this.minNode_ = node.parent;
|
||||||
|
this.balance_(node.parent);
|
||||||
|
} else if (node.isRightChild()) {
|
||||||
|
node.parent.right = null;
|
||||||
|
if (node == this.maxNode_) this.maxNode_ = node.parent;
|
||||||
|
this.balance_(node.parent);
|
||||||
|
} else {
|
||||||
|
this.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -651,21 +673,21 @@ AvlTree.prototype.removeNode_ = function(node) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getMinNode_ = function(opt_rootNode) {
|
AvlTree.prototype.getMinNode_ = function(opt_rootNode) {
|
||||||
if (!opt_rootNode) {
|
if (!opt_rootNode) {
|
||||||
return this.minNode_;
|
return this.minNode_;
|
||||||
}
|
|
||||||
|
|
||||||
var minNode = opt_rootNode;
|
|
||||||
this.traverse_(function(node) {
|
|
||||||
var retNode = null;
|
|
||||||
if (node.left) {
|
|
||||||
minNode = node.left;
|
|
||||||
retNode = node.left;
|
|
||||||
}
|
}
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
}, opt_rootNode);
|
|
||||||
|
|
||||||
return minNode;
|
var minNode = opt_rootNode;
|
||||||
|
this.traverse_(function(node) {
|
||||||
|
var retNode = null;
|
||||||
|
if (node.left) {
|
||||||
|
minNode = node.left;
|
||||||
|
retNode = node.left;
|
||||||
|
}
|
||||||
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
|
}, opt_rootNode);
|
||||||
|
|
||||||
|
return minNode;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -679,21 +701,21 @@ AvlTree.prototype.getMinNode_ = function(opt_rootNode) {
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
AvlTree.prototype.getMaxNode_ = function(opt_rootNode) {
|
AvlTree.prototype.getMaxNode_ = function(opt_rootNode) {
|
||||||
if (!opt_rootNode) {
|
if (!opt_rootNode) {
|
||||||
return this.maxNode_;
|
return this.maxNode_;
|
||||||
}
|
|
||||||
|
|
||||||
var maxNode = opt_rootNode;
|
|
||||||
this.traverse_(function(node) {
|
|
||||||
var retNode = null;
|
|
||||||
if (node.right) {
|
|
||||||
maxNode = node.right;
|
|
||||||
retNode = node.right;
|
|
||||||
}
|
}
|
||||||
return retNode; // If null, we'll stop traversing the tree
|
|
||||||
}, opt_rootNode);
|
|
||||||
|
|
||||||
return maxNode;
|
var maxNode = opt_rootNode;
|
||||||
|
this.traverse_(function(node) {
|
||||||
|
var retNode = null;
|
||||||
|
if (node.right) {
|
||||||
|
maxNode = node.right;
|
||||||
|
retNode = node.right;
|
||||||
|
}
|
||||||
|
return retNode; // If null, we'll stop traversing the tree
|
||||||
|
}, opt_rootNode);
|
||||||
|
|
||||||
|
return maxNode;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -708,19 +730,19 @@ AvlTree.prototype.getMaxNode_ = function(opt_rootNode) {
|
||||||
* @constructor
|
* @constructor
|
||||||
*/
|
*/
|
||||||
AvlTree.Node = function(value, opt_parent) {
|
AvlTree.Node = function(value, opt_parent) {
|
||||||
/**
|
/**
|
||||||
* The value stored by the node.
|
* The value stored by the node.
|
||||||
*
|
*
|
||||||
* @type {*}
|
* @type {*}
|
||||||
*/
|
*/
|
||||||
this.value = value;
|
this.value = value;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The node's parent. Null if the node is the root.
|
* The node's parent. Null if the node is the root.
|
||||||
*
|
*
|
||||||
* @type {AvlTree.Node}
|
* @type {AvlTree.Node}
|
||||||
*/
|
*/
|
||||||
this.parent = opt_parent ? opt_parent : null;
|
this.parent = opt_parent ? opt_parent : null;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -756,7 +778,7 @@ AvlTree.Node.prototype.height = 1;
|
||||||
* child of its parent.
|
* child of its parent.
|
||||||
*/
|
*/
|
||||||
AvlTree.Node.prototype.isRightChild = function() {
|
AvlTree.Node.prototype.isRightChild = function() {
|
||||||
return !!this.parent && this.parent.right == this;
|
return !!this.parent && this.parent.right == this;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -768,5 +790,5 @@ AvlTree.Node.prototype.isRightChild = function() {
|
||||||
* child of its parent.
|
* child of its parent.
|
||||||
*/
|
*/
|
||||||
AvlTree.Node.prototype.isLeftChild = function() {
|
AvlTree.Node.prototype.isLeftChild = function() {
|
||||||
return !!this.parent && this.parent.left == this;
|
return !!this.parent && this.parent.left == this;
|
||||||
};
|
};
|
Loading…
Reference in New Issue
Block a user