-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbinaryHeap.js
147 lines (132 loc) · 4.41 KB
/
binaryHeap.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
/**
* @author Derrick Burns
*/
binaryHeap = function (comparisonFunction) {
this.cmp = comparisonFunction;
this.content = [];
};
binaryHeap.prototype = {
content: [],
toArray: function( ) {
var r = "[";
for( var i=0; i!= this.content.length; i++ ) {
r = r + this.content[i].toString();
r = r + ", ";
}
r = r + "]";
return r;
},
push: function (element) {
// Add the new element to the end of the array.
this.content.push(element);
// Allow it to sink down.
this.sinkDown(this.content.length - 1);
},
pop: function () {
// Store the first element so we can return it later.
var result = this.content[0];
// Get the element at the end of the array.
var end = this.content.pop();
// If there are any elements left, put the end element at the
// start, and let it bubble up.
if (this.content.length > 0) {
this.content[0] = end;
this.bubbleUp(0);
}
return result;
},
remove: function (node) {
var len = this.content.length;
// To remove a value, we must search through the array to find
// it.
for (var i = 0; i < len; i++) {
if (this.content[i] === node) {
// When it is found, the process seen in 'pop' is repeated
// to fill up the hole.
var end = this.content.pop();
if (i !== this.content.length - 1) {
this.content[i] = end;
this.bubbleUp(i);
}
return;
}
}
throw new Error("Node not found.");
},
size: function () {
return this.content.length;
},
sinkDown: function (n) {
// Fetch the element that has to be sunk.
var element = this.content[n];
// When at 0, an element can not sink any further.
while (n > 0) {
// Compute the parent element's index, and fetch it.
var parentN = Math.floor((n + 1) / 2) - 1,
parent = this.content[parentN];
// Swap the elements if the parent is greater.
if (this.cmp(element, parent) < 0) {
this.content[parentN] = element;
this.content[n] = parent;
// Update 'n' to continue at the new position.
n = parentN;
}
// Found a parent that is less, no need to sink any further.
else {
break;
}
}
},
bubbleUp: function (n) {
// Look up the target element.
var length = this.content.length,
element = this.content[n];
while (true) {
// Compute the indices of the child elements.
var child2N = (n + 1) * 2,
child1N = child2N - 1;
// This is used to store the new position of the element,
// if any.
var swap = null;
// If the first child exists (is inside the array)...
if (child1N < length) {
// Look it up
var child1 = this.content[child1N];
// If it is less than our element's, we need to swap.
if (this.cmp(child1, element) < 0) {
swap = child1N;
}
}
// Do the same checks for the other child.
if (child2N < length) {
var child2 = this.content[child2N];
if (this.cmp(child2, (swap === null ? element: child1)) < 0) {
swap = child2N;
}
}
// If the element needs to be moved, swap it, and continue.
if (swap !== null) {
this.content[n] = this.content[swap];
this.content[swap] = element;
n = swap;
}
// Otherwise, we are done.
else {
break;
}
}
}
};
BinaryHeap = function(comparisonFunction) {
var b = new binaryHeap(comparisonFunction);
return {
toArray: function() { return b.toArray(); },
push: function(elem) { return b.push(elem); },
pop: function() { return b.pop(); },
remove: function(elem) { return b.remove(elem); },
size: function() { return b.size(); }
}
}();
exports = {
BinaryHeap: BinaryHeap
};