You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
216 lines
3.9 KiB
216 lines
3.9 KiB
/**
|
|
* Mnemonist Queue
|
|
* ================
|
|
*
|
|
* Queue implementation based on the ideas of Queue.js that seems to beat
|
|
* a LinkedList one in performance.
|
|
*/
|
|
var Iterator = require('obliterator/iterator'),
|
|
forEach = require('obliterator/foreach');
|
|
|
|
/**
|
|
* Queue
|
|
*
|
|
* @constructor
|
|
*/
|
|
function Queue() {
|
|
this.clear();
|
|
}
|
|
|
|
/**
|
|
* Method used to clear the queue.
|
|
*
|
|
* @return {undefined}
|
|
*/
|
|
Queue.prototype.clear = function() {
|
|
|
|
// Properties
|
|
this.items = [];
|
|
this.offset = 0;
|
|
this.size = 0;
|
|
};
|
|
|
|
/**
|
|
* Method used to add an item to the queue.
|
|
*
|
|
* @param {any} item - Item to enqueue.
|
|
* @return {number}
|
|
*/
|
|
Queue.prototype.enqueue = function(item) {
|
|
|
|
this.items.push(item);
|
|
return ++this.size;
|
|
};
|
|
|
|
/**
|
|
* Method used to retrieve & remove the first item of the queue.
|
|
*
|
|
* @return {any}
|
|
*/
|
|
Queue.prototype.dequeue = function() {
|
|
if (!this.size)
|
|
return;
|
|
|
|
var item = this.items[this.offset];
|
|
|
|
if (++this.offset * 2 >= this.items.length) {
|
|
this.items = this.items.slice(this.offset);
|
|
this.offset = 0;
|
|
}
|
|
|
|
this.size--;
|
|
|
|
return item;
|
|
};
|
|
|
|
/**
|
|
* Method used to retrieve the first item of the queue.
|
|
*
|
|
* @return {any}
|
|
*/
|
|
Queue.prototype.peek = function() {
|
|
if (!this.size)
|
|
return;
|
|
|
|
return this.items[this.offset];
|
|
};
|
|
|
|
/**
|
|
* Method used to iterate over the queue.
|
|
*
|
|
* @param {function} callback - Function to call for each item.
|
|
* @param {object} scope - Optional scope.
|
|
* @return {undefined}
|
|
*/
|
|
Queue.prototype.forEach = function(callback, scope) {
|
|
scope = arguments.length > 1 ? scope : this;
|
|
|
|
for (var i = this.offset, j = 0, l = this.items.length; i < l; i++, j++)
|
|
callback.call(scope, this.items[i], j, this);
|
|
};
|
|
|
|
/*
|
|
* Method used to convert the queue to a JavaScript array.
|
|
*
|
|
* @return {array}
|
|
*/
|
|
Queue.prototype.toArray = function() {
|
|
return this.items.slice(this.offset);
|
|
};
|
|
|
|
/**
|
|
* Method used to create an iterator over a queue's values.
|
|
*
|
|
* @return {Iterator}
|
|
*/
|
|
Queue.prototype.values = function() {
|
|
var items = this.items,
|
|
i = this.offset;
|
|
|
|
return new Iterator(function() {
|
|
if (i >= items.length)
|
|
return {
|
|
done: true
|
|
};
|
|
|
|
var value = items[i];
|
|
i++;
|
|
|
|
return {
|
|
value: value,
|
|
done: false
|
|
};
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Method used to create an iterator over a queue's entries.
|
|
*
|
|
* @return {Iterator}
|
|
*/
|
|
Queue.prototype.entries = function() {
|
|
var items = this.items,
|
|
i = this.offset,
|
|
j = 0;
|
|
|
|
return new Iterator(function() {
|
|
if (i >= items.length)
|
|
return {
|
|
done: true
|
|
};
|
|
|
|
var value = items[i];
|
|
i++;
|
|
|
|
return {
|
|
value: [j++, value],
|
|
done: false
|
|
};
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Attaching the #.values method to Symbol.iterator if possible.
|
|
*/
|
|
if (typeof Symbol !== 'undefined')
|
|
Queue.prototype[Symbol.iterator] = Queue.prototype.values;
|
|
|
|
/**
|
|
* Convenience known methods.
|
|
*/
|
|
Queue.prototype.toString = function() {
|
|
return this.toArray().join(',');
|
|
};
|
|
|
|
Queue.prototype.toJSON = function() {
|
|
return this.toArray();
|
|
};
|
|
|
|
Queue.prototype.inspect = function() {
|
|
var array = this.toArray();
|
|
|
|
// Trick so that node displays the name of the constructor
|
|
Object.defineProperty(array, 'constructor', {
|
|
value: Queue,
|
|
enumerable: false
|
|
});
|
|
|
|
return array;
|
|
};
|
|
|
|
if (typeof Symbol !== 'undefined')
|
|
Queue.prototype[Symbol.for('nodejs.util.inspect.custom')] = Queue.prototype.inspect;
|
|
|
|
/**
|
|
* Static @.from function taking an arbitrary iterable & converting it into
|
|
* a queue.
|
|
*
|
|
* @param {Iterable} iterable - Target iterable.
|
|
* @return {Queue}
|
|
*/
|
|
Queue.from = function(iterable) {
|
|
var queue = new Queue();
|
|
|
|
forEach(iterable, function(value) {
|
|
queue.enqueue(value);
|
|
});
|
|
|
|
return queue;
|
|
};
|
|
|
|
/**
|
|
* Static @.of function taking an arbitrary number of arguments & converting it
|
|
* into a queue.
|
|
*
|
|
* @param {...any} args
|
|
* @return {Queue}
|
|
*/
|
|
Queue.of = function() {
|
|
return Queue.from(arguments);
|
|
};
|
|
|
|
/**
|
|
* Exporting.
|
|
*/
|
|
module.exports = Queue;
|