gristlabs_grist-core/app/common/BinaryIndexedTree.js

259 lines
7.5 KiB
JavaScript
Raw Permalink Normal View History

/**
* Implements a binary indexed tree, aka Fenwick tree. See
* http://en.wikipedia.org/wiki/Fenwick_tree
*/
function BinaryIndexedTree(optSize) {
this.tree = [];
if (optSize > 0) {
this.tree.length = optSize;
for (var i = 0; i < optSize; i++) {
this.tree[i] = 0;
}
// The last valid index rounded down to the nearest power of 2.
this.mask = mostSignificantOne(this.tree.length - 1);
}
}
/**
* Returns a number that contains only the least significant one in `num`.
* @param {Number} num - Positive integer.
* @returns {Number} The least significant one in `num`, e.g. for 10110, returns 00010.
*/
function leastSignificantOne(num) {
return num & (-num);
}
BinaryIndexedTree.leastSignificantOne = leastSignificantOne;
/**
* Strips the least significant one from `num`.
* @param {Number} num - Positive integer.
* @returns {Number} `num` with the least significant one removed, e.g. for 10110, returns 10100.
*/
function stripLeastSignificantOne(num) {
return num & (num - 1);
}
BinaryIndexedTree.stripLeastSignificantOne = stripLeastSignificantOne;
function mostSignificantOne(num) {
if (num === 0) {
return 0;
}
var msb = 1;
while ((num >>>= 1)) {
msb <<= 1;
}
return msb;
}
BinaryIndexedTree.mostSignificantOne = mostSignificantOne;
/**
* Converts in-place an array of cumulative values to the original values.
* @param {Array<number>} values - Array of cumulative values, or partial sums.
* @returns {Array<number>} - same `values` array, with elements replaced by deltas.
* E.g. [1,3,6,10] is converted to [1,2,3,4].
*/
function cumulToValues(values) {
for (var i = values.length - 1; i >= 1; i--) {
values[i] -= values[i - 1];
}
return values;
}
BinaryIndexedTree.cumulToValues = cumulToValues;
/**
* Converts in-place an array of values to cumulative values, or partial sums.
* @param {Array<number>} values - Array of numerical values.
* @returns {Array<number>} - same `values` array, with elements replaced by partial sums.
* E.g. [1,2,3,4] is converted to [1,3,6,10].
*/
function valuesToCumul(values) {
for (var i = 1; i < values.length; i++) {
values[i] += values[i - 1];
}
return values;
}
BinaryIndexedTree.valuesToCumul = valuesToCumul;
/**
* @returns {Number} length of the tree.
*/
BinaryIndexedTree.prototype.size = function() {
return this.tree.length;
};
/**
* Converts the BinaryIndexedTree to a cumulative array.
* Takes time linear in the size of the array.
* @returns {Array<number>} - array with each element a partial sum.
*/
BinaryIndexedTree.prototype.toCumulativeArray = function() {
var cumulValues = [this.tree[0]];
var len = cumulValues.length = this.tree.length;
for (var i = 1; i < len; i++) {
cumulValues[i] = this.tree[i] + cumulValues[stripLeastSignificantOne(i)];
}
return cumulValues;
};
/**
* Converts the BinaryIndexedTree to an array of individual values.
* Takes time linear in the size of the array.
* @returns {Array<number>} - array with each element containing the value that was inserted.
*/
BinaryIndexedTree.prototype.toValueArray = function() {
return cumulToValues(this.toCumulativeArray());
};
/**
* Creates a tree from an array of cumulative values.
* Takes time linear in the size of the array.
* @param {Array<number>} - array with each element a partial sum.
*/
BinaryIndexedTree.prototype.fillFromCumulative = function(cumulValues) {
var len = this.tree.length = cumulValues.length;
if (len > 0) {
this.tree[0] = cumulValues[0];
for (var i = 1; i < len; i++) {
this.tree[i] = cumulValues[i] - cumulValues[stripLeastSignificantOne(i)];
}
// The last valid index rounded down to the nearest power of 2.
this.mask = mostSignificantOne(this.tree.length - 1);
} else {
this.mask = 0;
}
};
/**
2022-02-19 09:46:49 +00:00
* Creates a tree from an array of individual values.
* Takes time linear in the size of the array.
* @param {Array<number>} - array with each element containing the value to insert.
*/
BinaryIndexedTree.prototype.fillFromValues = function(values) {
this.fillFromCumulative(valuesToCumul(values.slice()));
};
/**
* Reads the cumulative value at the given index. Takes time O(log(index)).
* @param {Number} index - index in the array.
* @returns {Number} - cumulative values up to and including `index`.
*/
BinaryIndexedTree.prototype.getCumulativeValue = function(index) {
var sum = this.tree[0];
while (index > 0) {
sum += this.tree[index];
index = stripLeastSignificantOne(index);
}
return sum;
};
/**
* Reads the cumulative value from start(inclusive) to end(exclusive). Takes time O(log(end)).
* @param {Number} start - start index
* @param {Number} end - end index
* @returns {Number} - cumulative values between start(inclusive) and end(exclusive)
*/
BinaryIndexedTree.prototype.getCumulativeValueRange = function(start, end) {
return this.getSumTo(end) - this.getSumTo(start);
};
/**
* Returns the sum of values up to the given index. Takes time O(log(index)).
* @param {Number} index - index in the array.
* @returns {Number} - cumulative values up to but not including `index`.
*/
BinaryIndexedTree.prototype.getSumTo = function(index) {
return (index > 0 ? this.getCumulativeValue(index - 1) : 0);
};
/**
* Returns the total of all values in the tree. Takes time O(log(N)).
* @returns {Number} - sum of all values.
*/
BinaryIndexedTree.prototype.getTotal = function() {
return this.getCumulativeValue(this.tree.length - 1);
};
/**
* Reads a single value at the given index. Takes time O(log(index)).
* @param {Number} index - index in the array.
* @returns {Number} - the value that was inserted at `index`.
*/
BinaryIndexedTree.prototype.getValue = function(index) {
var value = this.tree[index];
if (index > 0) {
var parent = stripLeastSignificantOne(index);
index--;
while (index !== parent) {
value -= this.tree[index];
index = stripLeastSignificantOne(index);
}
}
return value;
};
/**
* Updates a value at an index. Takes time O(log(table size)).
* @param {Number} index - index in the array.
* @param {Number} delta - value to add to the previous value at `index`.
*/
BinaryIndexedTree.prototype.addValue = function(index, delta) {
if (index === 0) {
this.tree[0] += delta;
} else {
while (index < this.tree.length) {
this.tree[index] += delta;
index += leastSignificantOne(index);
}
}
};
/**
* Sets a value at an index. Takes time O(log(table size)).
* @param {Number} index - index in the array.
* @param {Number} value - new value to set at `index`.
*/
BinaryIndexedTree.prototype.setValue = function(index, value) {
this.addValue(index, value - this.getValue(index));
};
/**
* Given a cumulative value, finds the first element whose inclusion reaches the value.
* E.g. for values [1,2,3,4] (cumulative [1,3,6,10]), getIndex(3) = 1, getIndex(3.1) = 2.
* @param {Number} cumulValue - cumulative value to exceed.
* @returns {Number} index - the first index such that getCumulativeValue(index) >= cumulValue.
* If cumulValue is too large, return one more than the highest valid index.
*/
BinaryIndexedTree.prototype.getIndex = function(cumulValue) {
if (this.tree.length === 0 || this.tree[0] >= cumulValue) {
return 0;
}
var index = 0;
var mask = this.mask;
var sum = this.tree[0];
while (mask !== 0) {
var testIndex = index + mask;
if (testIndex < this.tree.length && sum + this.tree[testIndex] < cumulValue) {
index = testIndex;
sum += this.tree[index];
}
mask >>>= 1;
}
return index + 1;
};
module.exports = BinaryIndexedTree;