com.opengamma.strata.math.impl.statistics.descriptive.QuantileCalculationMethod Maven / Gradle / Ivy
/*
* Copyright (C) 2015 - present by OpenGamma Inc. and the OpenGamma group of companies
*
* Please see distribution for license.
*/
package com.opengamma.strata.math.impl.statistics.descriptive;
import com.opengamma.strata.collect.ArgChecker;
import com.opengamma.strata.collect.array.DoubleArray;
/**
* Abstract method to estimate quantiles and expected shortfalls from sample observations.
*/
public abstract class QuantileCalculationMethod {
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, the quantile estimation with the level 99% corresponds to
* the smallest 99% observations and 1% of the observation are above that level.
*
* If index value computed from the level is outside of the sample data range,
* {@code IllegalArgumentException} is thrown.
*
* The sample observations are supposed to be unsorted.
*
* The quantile result produced contains the quantile value, the indices of the data points used to compute
* it as well as the weights assigned to each point in the computation. The indices are based on the original,
* unsorted array. Additionally, the indices start from 0 and so do not need to be shifted to account for java
* indexing, when using them to reference the data points in the quantile calculation.
*
* @param level the quantile level
* @param sample the sample observations
* @return the quantile estimation
*/
public QuantileResult quantileResultFromUnsorted(double level, DoubleArray sample) {
return quantile(level, sample, false);
}
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, the quantile estimation with the level 99% corresponds to
* the smallest 99% observations and 1% of the observation are above that level.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* quantile is computed with flat extrapolation.
*
* The sample observations are supposed to be unsorted.
*
* The quantile result produced contains the quantile value, the indices of the data points used to compute
* it as well as the weights assigned to each point in the computation. The indices are based on the original,
* unsorted array. Additionally, the indices start from 0 and so do not need to be shifted to account for java
* indexing, when using them to reference the data points in the quantile calculation.
*
* @param level the quantile level
* @param sample the sample observations
* @return the quantile estimation
*/
public QuantileResult quantileResultWithExtrapolationFromUnsorted(double level, DoubleArray sample) {
return quantile(level, sample, true);
}
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the quantile estimation with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range,
* {@code IllegalArgumentException} is thrown.
*
* The sample observations are sorted from the smallest to the largest.
*
* @param level the quantile level
* @param sortedSample the sample observations
* @return the quantile estimation
*/
public double quantileFromSorted(double level, DoubleArray sortedSample) {
return quantileResultFromUnsorted(level, sortedSample).getValue();
}
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the quantile estimation with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range,
* {@code IllegalArgumentException} is thrown.
*
* The sample observations are supposed to be unsorted, the first step is to sort the data.
*
* @param level the quantile level
* @param sample the sample observations
* @return The quantile estimation
*/
public double quantileFromUnsorted(double level, DoubleArray sample) {
return quantileFromSorted(level, sample.sorted());
}
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the quantile estimation with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* quantile is computed with flat extrapolation.
*
* The sample observations are sorted from the smallest to the largest.
*
* @param level the quantile level
* @param sortedSample the sample observations
* @return the quantile estimation
*/
public double quantileWithExtrapolationFromSorted(double level, DoubleArray sortedSample) {
return quantileResultWithExtrapolationFromUnsorted(level, sortedSample).getValue();
}
/**
* Compute the quantile estimation.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the quantile estimation with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* quantile is computed with flat extrapolation.
*
* The sample observations are supposed to be unsorted, the first step is to sort the data.
*
* @param level the quantile level
* @param sample the sample observations
* @return The quantile estimation
*/
public double quantileWithExtrapolationFromUnsorted(double level, DoubleArray sample) {
return quantileWithExtrapolationFromSorted(level, sample.sorted());
}
//-------------------------------------------------------------------------
/**
* Compute the expected shortfall.
*
* The shortfall level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, the expected shortfall with the level 99% corresponds to
* the average of the smallest 99% of the observations.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* expected short fall is computed with flat extrapolation.
* Thus this is coherent to {@link #quantileWithExtrapolationFromUnsorted(double, DoubleArray)}.
*
* The sample observations are supposed to be unsorted.
*
* The quantile result produced contains the expected shortfall value, the indices of the data points used to compute
* it as well as the weights assigned to each point in the computation. The indices are based on the original,
* unsorted array. Additionally, the indices start from 0 and so do not need to be shifted to account for java
* indexing, when using them to reference the data points in the quantile calculation.
*
* @param level the quantile level
* @param sample the sample observations
* @return the quantile estimation
*/
public QuantileResult expectedShortfallResultFromUnsorted(double level, DoubleArray sample) {
return expectedShortfall(level, sample);
}
/**
* Compute the expected shortfall.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the expected shortfall with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* expected short fall is computed with flat extrapolation.
* Thus this is coherent to {@link #quantileWithExtrapolationFromSorted(double, DoubleArray)}.
*
* The sample observations are sorted from the smallest to the largest.
*
* @param level the quantile level
* @param sortedSample the sample observations
* @return the quantile estimation
*/
public double expectedShortfallFromSorted(double level, DoubleArray sortedSample) {
return expectedShortfallResultFromUnsorted(level, sortedSample).getValue();
}
/**
* Compute the expected shortfall.
*
* The quantile level is in decimal, i.e. 99% = 0.99 and 0 < level < 1 should be satisfied.
* This is measured from the bottom, that is, Thus the expected shortfall with the level 99% corresponds to
* the smallest 99% observations.
*
* If index value computed from the level is outside of the sample data range, the nearest data point is used, i.e.,
* expected short fall is computed with flat extrapolation.
* Thus this is coherent to {@link #quantileWithExtrapolationFromUnsorted(double, DoubleArray)}.
*
* The sample observations are supposed to be unsorted, the first step is to sort the data.
*
* @param level the quantile level
* @param sample the sample observations
* @return The expected shortfall estimation
*/
public double expectedShortfallFromUnsorted(double level, DoubleArray sample) {
return expectedShortfallFromSorted(level, sample.sorted());
}
//-------------------------------------------------------------------------
/**
* Computed the quantile.
*
* This protected method should be implemented in subclasses.
*
* @param level the quantile level
* @param sample the sample observations
* @param isExtrapolated extrapolated if true, not extrapolated otherwise.
* @return the quantile
*/
protected abstract QuantileResult quantile(double level, DoubleArray sample, boolean isExtrapolated);
/**
* Computed the expected shortfall.
*
* This protected method should be implemented in subclasses
* and coherent to {@link #quantile(double, DoubleArray, boolean)}.
*
* @param level the quantile level
* @param sample the sample observations
* @return the expected shortfall
*/
protected abstract QuantileResult expectedShortfall(double level, DoubleArray sample);
/**
* Check the index is within the sample data range.
*
* If the index is outside the data range, the nearest data point is used in case of {@code isExtrapolated == true} or
* an exception is thrown in case of {@code isExtrapolated == false}.
*
* @param index the index
* @param size the sample size
* @param isExtrapolated extrapolated if true, not extrapolated otherwise
* @return the index
*/
protected double checkIndex(double index, int size, boolean isExtrapolated) {
if (isExtrapolated) {
return Math.min(Math.max(index, 1), size);
}
ArgChecker.isTrue(index >= 1, "Quantile can not be computed below the lowest probability level.");
ArgChecker.isTrue(index <= size, "Quantile can not be computed above the highest probability level.");
return index;
}
}