All Downloads are FREE. Search and download functionalities are using the official Maven repository.

tech.tablesaw.analytic.AggregateFunctions Maven / Gradle / Ivy

There is a newer version: 0.43.1
Show newest version
package tech.tablesaw.analytic;

import java.util.ArrayDeque;
import java.util.function.Function;
import tech.tablesaw.analytic.WindowFrame.WindowGrowthType;
import tech.tablesaw.api.ColumnType;
import tech.tablesaw.columns.numbers.DoubleColumnType;

/**
 * Analytic Aggregate functions.
 *
 * 

Analytic Aggregate functions require different implementations compared to regular aggregate * functions because they are called once per row and must return a value for every row in the * table. * *

Consider calculating the SUM over a table with a window definition of ROWS BETWEEN UNBOUNDED * PRECEDING AND CURRENT ROW. If a regular aggregate function was used it would be called once for * each row and since each window contains O(n) rows for a total running time of O(n^2). Clearly you * can use a more efficient algorithm that keeps a running sum as rows are added to the window and * runs in O(n). This class allows for those more efficient algorithms to be used. * *

If at least one side of the window is unbounded the window is considered an append window. * With a bit of tweaking windows UNBOUNDED FOLLOWING windows can be converted to UNBOUNDED * PRECEDING windows so they are append only and can use a more efficient algorithm similar to the * one explained above. * *

Sliding windows are windows where both sides of the window are following, preceding or current * row. Analytic aggregate algorithms for sliding windows are generally implemented with a {@link * java.util.Deque} so that elements can be added or removed from either side of the window as it * slides. * *

This class creates two implementations per analytic aggregate function. One for append windows * and one for sliding windows. */ enum AggregateFunctions implements FunctionMetaData { SUM(new Sum<>(), ColumnType.DOUBLE, AggregateFunctions::isNumericColumn), MAX(new Max<>(), ColumnType.DOUBLE, AggregateFunctions::isNumericColumn), MIN(new Min<>(), ColumnType.DOUBLE, AggregateFunctions::isNumericColumn), MEAN(new Mean<>(), ColumnType.DOUBLE, AggregateFunctions::isNumericColumn), COUNT(new Count<>(), ColumnType.INTEGER, t -> true); private final WindowDependentAggregateFunction implementation; private final ColumnType outputColumnType; private final Function isCompatibleColumnTestFunc; AggregateFunctions( WindowDependentAggregateFunction implementation, ColumnType outputColumnType, Function isCompatibleColumnTestFunc) { this.implementation = implementation; this.outputColumnType = outputColumnType; this.isCompatibleColumnTestFunc = isCompatibleColumnTestFunc; } /** Get the right implementation for that window definition. */ AggregateFunction getImplementation(WindowGrowthType growthType) { return this.implementation.functionFor(growthType); } @Override public String functionName() { return name(); } @Override public ColumnType returnType() { return outputColumnType; } @Override public boolean isCompatibleColumn(ColumnType type) { return isCompatibleColumnTestFunc.apply(type); } private static boolean isNumericColumn(ColumnType type) { return type.equals(ColumnType.DOUBLE) || type.equals(ColumnType.FLOAT) || type.equals(ColumnType.INTEGER) || type.equals(ColumnType.SHORT) || type.equals(ColumnType.LONG); } private abstract static class WindowDependentAggregateFunction { /** Sub classes of append windows should never call removeLeft. */ abstract static class AppendAggregateFunction implements AggregateFunction { @Override public final void removeLeftMost() { throw new UnsupportedOperationException( "Implementers of append aggregate functions should never call removeLeftMost"); } } abstract AppendAggregateFunction functionForAppendWindows(); abstract AggregateFunction functionForSlidingWindows(); AggregateFunction functionFor(WindowGrowthType growthType) { switch (growthType) { case FIXED: case FIXED_LEFT: case FIXED_RIGHT: return functionForAppendWindows(); case SLIDING: return functionForSlidingWindows(); } throw new IllegalArgumentException("Unexpected growthType: " + growthType); } } static class Sum extends WindowDependentAggregateFunction { @Override AppendAggregateFunction functionForAppendWindows() { return new AppendAggregateFunction() { private double sum = DoubleColumnType.missingValueIndicator(); @Override public Double getValue() { return sum; } @Override public void addRightMostMissing() {} @Override public void addRightMost(T newValue) { if (DoubleColumnType.valueIsMissing(sum)) { this.sum = 0.0; } this.sum += newValue.doubleValue(); } }; } @Override AggregateFunction functionForSlidingWindows() { return new AggregateFunction() { private final ArrayDeque queue = new ArrayDeque<>(); private Double sum = 0.0; private int missingCount = 0; @Override public void removeLeftMost() { Double removed = queue.remove(); if (DoubleColumnType.valueIsMissing(removed)) { missingCount--; } else { this.sum -= removed; } } @Override public void addRightMost(T newValue) { Double doubleValue = newValue.doubleValue(); this.sum += doubleValue; queue.add(doubleValue); } @Override public void addRightMostMissing() { queue.add(DoubleColumnType.missingValueIndicator()); missingCount++; } @Override public Double getValue() { if (queue.isEmpty() || missingCount == queue.size()) { return DoubleColumnType.missingValueIndicator(); } return sum; } }; } } static class Max extends WindowDependentAggregateFunction { @Override AppendAggregateFunction functionForAppendWindows() { return new AppendAggregateFunction() { private Double max = DoubleColumnType.missingValueIndicator(); @Override public void addRightMost(T newValue) { if (DoubleColumnType.valueIsMissing(max)) { max = newValue.doubleValue(); return; } this.max = Math.max(max, newValue.doubleValue()); } @Override public void addRightMostMissing() {} @Override public Double getValue() { return max; } }; } @Override AggregateFunction functionForSlidingWindows() { return new AggregateFunction() { private final ArrayDeque queue = new ArrayDeque<>(); @Override public void removeLeftMost() { queue.remove(); } @Override public void addRightMost(T newValue) { queue.add(newValue.doubleValue()); } @Override public void addRightMostMissing() { queue.add(DoubleColumnType.missingValueIndicator()); } @Override public Double getValue() { // This could be faster, but probably does not matter in practice because sliding windows // will be small. return queue.stream() .filter(d -> !DoubleColumnType.valueIsMissing(d)) .mapToDouble(Number::doubleValue) .max() .orElse(DoubleColumnType.missingValueIndicator()); } }; } } static class Min extends WindowDependentAggregateFunction { @Override AppendAggregateFunction functionForAppendWindows() { return new AppendAggregateFunction() { private Double min = DoubleColumnType.missingValueIndicator(); @Override public void addRightMost(T newValue) { if (DoubleColumnType.valueIsMissing(min)) { min = newValue.doubleValue(); return; } this.min = Math.min(min, newValue.doubleValue()); } @Override public void addRightMostMissing() {} @Override public Double getValue() { return min; } }; } @Override AggregateFunction functionForSlidingWindows() { return new AggregateFunction() { private final ArrayDeque queue = new ArrayDeque<>(); @Override public void removeLeftMost() { queue.remove(); } @Override public void addRightMost(T newValue) { queue.add(newValue.doubleValue()); } @Override public void addRightMostMissing() { queue.add(DoubleColumnType.missingValueIndicator()); } @Override public Double getValue() { // This could be faster, but probably does not matter in practice because sliding windows // will be small. return queue.stream() .filter(d -> !DoubleColumnType.valueIsMissing(d)) .mapToDouble(Number::doubleValue) .min() .orElse(DoubleColumnType.missingValueIndicator()); } }; } } static class Mean extends WindowDependentAggregateFunction { @Override AppendAggregateFunction functionForAppendWindows() { return new AppendAggregateFunction() { private double sum = DoubleColumnType.missingValueIndicator(); private double count = 0; @Override public Double getValue() { if (count == 0) { return DoubleColumnType.missingValueIndicator(); } return sum / count; } @Override public void addRightMostMissing() {} @Override public void addRightMost(T newValue) { if (DoubleColumnType.valueIsMissing(sum)) { this.sum = 0.0; } this.sum += newValue.doubleValue(); count++; } }; } @Override AggregateFunction functionForSlidingWindows() { return new AggregateFunction() { private final ArrayDeque queue = new ArrayDeque<>(); private Double sum = 0.0; private int missingCount = 0; @Override public void removeLeftMost() { Double removed = queue.remove(); if (DoubleColumnType.valueIsMissing(removed)) { missingCount--; } else { this.sum -= removed; } } @Override public void addRightMost(T newValue) { Double doubleValue = newValue.doubleValue(); this.sum += doubleValue; queue.add(doubleValue); } @Override public void addRightMostMissing() { queue.add(DoubleColumnType.missingValueIndicator()); missingCount++; } @Override public Double getValue() { if (queue.size() - missingCount == 0) { return DoubleColumnType.missingValueIndicator(); } return sum / (queue.size() - missingCount); } }; } } static class Count extends WindowDependentAggregateFunction { @Override AppendAggregateFunction functionForAppendWindows() { return new AppendAggregateFunction() { private int count = 0; @Override public Integer getValue() { return count; } @Override public void addRightMostMissing() {} @Override public void addRightMost(T newValue) { count++; } }; } @Override AggregateFunction functionForSlidingWindows() { return new AggregateFunction() { // Deque contains a boolean that when true indicates that the value in that position of the // window is missing. private final ArrayDeque queue = new ArrayDeque<>(); private int missingCount = 0; @Override public void removeLeftMost() { Boolean removedMissingValue = queue.remove(); if (removedMissingValue) { missingCount--; } } @Override public void addRightMost(T newValue) { queue.add(false); } @Override public void addRightMostMissing() { queue.add(true); missingCount++; } @Override public Integer getValue() { return queue.size() - missingCount; } }; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy