A binary heap, AKA priority queue, with quick access to min and max prioritary elements.
A binary heap is a data structure with 2 properties:
- Heap shape: The heap will have all levels, except the last one, fullfilled with max number of elements possible in this level. For example, see the image below:
As you can see, the first's 3 levels have all number of nodes possible in the respective level (
pow(base: 2,exponent: level)
):
-
Level 0: 1 node
-
Level 1: 2 nodes
-
Level 2: 4 nodes
-
...
-
In a traditional min-heap, all nodes that have descendants should be less than all your respective descendants. The image above explains for yourself.
An interest thing about binary heaps in general is the fact they can be implemented in a List of values, not necessary with linked list of nodes. This implementation uses 'list/array'.
If you try to find the maximum value in a min-heap, you will have O(lg n) in worst case. The same with find minimum value in a max-heap.
With MinMax heap, you have O(1) in the worst case to find both min and max value of all heap.
Operations like insert, remove (min and max) have O(lg n) in worst case. And the build method (when you pass a List<E>
values in the input parameter) have O(n lg n) in worst case.
Here an image to illustrate a valid MinMax heap:
As you see, the min will always be the root, and the max will be or the direct left child or the direct right child of the root.
You can build a heap from a List<E>
values in input parameter.
This approach is mantained for compatibility reasons.
void main() {
final myIntHeap = MinMaxHeap<int>(input: [1, 2, 3, 4, 5, 6, 7, 8]);
}
To build a heap from any iterable, use the factory constructor fromIterable
.
void main() {
final myIntHeap = MinMaxHeap.fromIterable(iterable: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
}
Or, you can build with successive insertions.
void main() {
const myInput = <int>[1, 2, 3, 4, 5, 6, 7, 8];
final myIntHeap = MinMaxHeap<int>();
myInput.forEach(myIntHeap.add);
// or ```myIntHeap.insert```
// same of loop trough myInput and call add method in every cycle.
}
Other aspect to consider is the use of the criteria parameter.
If you want to build your MinMax heap with custom criteria, you need to pass a callback funtion that returns num
or int
or double
types.
Else will throw an Error.
void main() {
const myInput = <String>['hello', 'my', 'name', 'is', 'Elan', 'and', 'your', 'name', '?'];
final myStringHeap = MinMaxHeap<String>.fromIterable(iterable: myInput, criteria: (word) => word.length);
}
You can use sucessive insertions with callbacks too.
void main() {
const myInput = <String>['hello', 'my', 'name', 'is', 'Elan', 'and', 'your', 'name', '?'];
final myStringHeap = MinMaxHeap<String>(criteria: (word) => word.length);
for(final element in myInput) {
myStringHeap.add(element);
}
}
Curiosity: If you use the same List of elements, but build one heap passing the iterable parameter and build other heap with the same iterable, but with sucessive insertions (add), the results heaps will be different! But both are valid ones. Test and see! To more details, see the test file and search the test "From a List, is a valid min-max heap?".
This package can be improved and tips are welcome! Feel free to create an issue in Github's repository and give a feedback.