Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package de.gsi.chart.renderer.datareduction;
import de.gsi.chart.renderer.RendererDataReducer;
import de.gsi.dataset.utils.ProcessingProfiler;
/**
* -- essentially a down-sampling data reduction algorithm
*
* @author braeun
*/
public class MaxDataReducer implements RendererDataReducer {
private static final int DEFAULT_MAX_POINTS_COUNT = 1000;
@Override
public int reducePoints(final double[] xValues, final double[] yValues, final double[] xPointErrorsPos,
final double[] xPointErrorsNeg, final double[] yPointErrorsPos, final double[] yPointErrorsNeg,
final String[] styles, final boolean[] pointSelected, final int indexMin, final int indexMax) {
final int size = indexMax - indexMin;
if (size <= MaxDataReducer.DEFAULT_MAX_POINTS_COUNT) {
// just shift the data set to front
MaxDataReducer.shiftDataToFront(xValues, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(yValues, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(xPointErrorsPos, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(xPointErrorsNeg, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(yPointErrorsPos, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(yPointErrorsNeg, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(styles, indexMin, indexMax);
MaxDataReducer.shiftDataToFront(pointSelected, indexMin, indexMax);
return size;
}
return reduce(xValues, yValues, xPointErrorsPos, xPointErrorsNeg, yPointErrorsPos, yPointErrorsNeg, styles,
pointSelected, indexMin, indexMax, MaxDataReducer.DEFAULT_MAX_POINTS_COUNT);
}
private static void shiftDataToFront(final double[] data, final int indexMin, final int indexMax) {
final int size = indexMax - indexMin;
System.arraycopy(data, indexMin, data, 0, size);
}
private static void shiftDataToFront(final boolean[] data, final int indexMin, final int indexMax) {
final int size = indexMax - indexMin;
System.arraycopy(data, indexMin, data, 0, size);
}
private static void shiftDataToFront(final String[] data, final int indexMin, final int indexMax) {
final int size = indexMax - indexMin;
System.arraycopy(data, indexMin, data, 0, size);
}
private static void moveInPlace(final double[] xValues, final double[] yValues, final double[] xPointErrorsPos,
final double[] xPointErrorsNeg, final double[] yPointErrorsPos, final double[] yPointErrorsNeg,
final String[] styles, final boolean[] pointSelected, final int fromIndex, final int toIndex) {
xValues[toIndex] = xValues[fromIndex];
yValues[toIndex] = yValues[fromIndex];
xPointErrorsPos[toIndex] = xPointErrorsPos[fromIndex];
xPointErrorsNeg[toIndex] = xPointErrorsNeg[fromIndex];
yPointErrorsPos[toIndex] = yPointErrorsPos[fromIndex];
yPointErrorsNeg[toIndex] = yPointErrorsNeg[fromIndex];
styles[toIndex] = styles[fromIndex];
pointSelected[toIndex] = pointSelected[fromIndex];
}
// TODO: check again algorithm with original implementation... some error was introduced
private int reduce(final double[] xValues, final double[] yValues, final double[] xPointErrorsPos,
final double[] xPointErrorsNeg, final double[] yPointErrorsPos, final double[] yPointErrorsNeg,
final String[] styles, final boolean[] pointSelected, final int indexMin, final int indexMax,
final int maxPointsCount) {
final long start = ProcessingProfiler.getTimeStamp();
final int size = indexMax - indexMin;
int count = 0;
int n = indexMin;
int nLastAdded = n;
MaxDataReducer.moveInPlace(xValues, yValues, xPointErrorsPos, xPointErrorsNeg, yPointErrorsPos, yPointErrorsNeg, styles,
pointSelected, n++, count++);
final double d = (double) size / (double) maxPointsCount;
int b = 1;
final int limit = (int) Math.round(b * d);
// difference w.r.t. last point
double delta = Math.abs(yValues[n] - yValues[count - 1]);
while (n < indexMax) {
if (n - nLastAdded >= limit) {
nLastAdded = n;
MaxDataReducer.moveInPlace(xValues, yValues, xPointErrorsPos, xPointErrorsNeg, yPointErrorsPos, yPointErrorsNeg,
styles, pointSelected, n++, count++);
// System.err.println(String.format("add point(%d)=%f",count,delta));
// yValues[count-1] += delta;
b++;
// limit = (int) Math.round(b * d);
} else {
final double delta1 = Math.abs(yValues[n] - yValues[count - 1]);
if (delta1 > delta) {
delta = delta1;
}
}
n++;
}
ProcessingProfiler.getTimeDiff(start,
String.format("data reduction (from %d to %d)", indexMax - indexMin, count));
return count;
}
/*
* old implementation
*
* @Override
* public List> reduce(ChartData data, Range dataRange, int
* maxPointsCount)
* {
* if (data.size() <= maxPointsCount)
* {
* List> list = new ArrayList<>();
* for (int i=0;i> reduced = new ArrayList<>(maxPointsCount);
* int n = 0;
* reduced.add(data.get(n++));
* double d = (double)data.size() / (double)maxPointsCount;
* int b = 1;
* int limit = (int)Math.round(b*d);
* XYChart.Data point = data.get(n++);
* double delta = Math.abs(point.getYValue().doubleValue() -
* reduced.get(reduced.size()-1).getYValue().doubleValue());
* while (n < data.size())
* {
* if (n >= limit)
* {
* reduced.add(point);
* point = data.get(n);
* b++;
* limit = (int)Math.round(b*d);
* }
* else
* {
* XYChart.Data p = data.get(n);
* double delta1 = Math.abs(p.getYValue().doubleValue() - reduced.get(reduced.size()-1).getYValue().doubleValue());
* if (delta1 > delta)
* {
* delta = delta1;
* point = p;
* }
* }
* n++;
* }
* return reduced;
* }
*/
}