Modified NormalDistributionImpl.cumulativeProbability to return 0 or 1,

respectively for values more than 40 standard deviations from the mean.
For these values, the actual probability is indistinguishable from 0 or 1
as a double.  Top coding improves performance for extreme values and prevents
convergence exceptions.

JIRA: MATH-414

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/branches/MATH_2_X@1040471 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Phil Steitz 2010-11-30 11:55:22 +00:00
parent 7cee480b68
commit 4185388b14
3 changed files with 27 additions and 17 deletions

View File

@ -171,25 +171,20 @@ public class NormalDistributionImpl extends AbstractContinuousDistribution
/**
* For this distribution, X, this method returns P(X &lt; <code>x</code>).
* If <code>x</code>is more than 40 standard deviations from the mean, 0 or 1 is returned,
* as in these cases the actual value is within <code>Double.MIN_VALUE</code> of 0 or 1.
*
* @param x the value at which the CDF is evaluated.
* @return CDF evaluated at <code>x</code>.
* @throws MathException if the algorithm fails to converge; unless
* x is more than 20 standard deviations from the mean, in which case the
* convergence exception is caught and 0 or 1 is returned.
* @throws MathException if the algorithm fails to converge
*/
public double cumulativeProbability(double x) throws MathException {
try {
return 0.5 * (1.0 + Erf.erf((x - mean) /
(standardDeviation * FastMath.sqrt(2.0))));
} catch (MaxIterationsExceededException ex) {
if (x < (mean - 20 * standardDeviation)) { // JDK 1.5 blows at 38
return 0.0d;
} else if (x > (mean + 20 * standardDeviation)) {
return 1.0d;
} else {
throw ex;
}
final double dev = x - mean;
if (FastMath.abs(dev) > 40 * standardDeviation) {
return dev < 0 ? 0.0d : 1.0d;
}
return 0.5 * (1.0 + Erf.erf((dev) /
(standardDeviation * FastMath.sqrt(2.0))));
}
/**

View File

@ -52,6 +52,13 @@ The <action> type attribute can be add,update,fix,remove.
If the output is not quite correct, check for invisible trailing spaces!
-->
<release version="2.2" date="TBD" description="TBD">
<action dev="psteitz" type="fix" issue="MATH-414">
Modified NormalDistributionImpl.cumulativeProbability to return 0 or 1,
respectively for values more than 40 standard deviations from the mean.
For these values, the actual probability is indistinguishable from 0 or 1
as a double. Top coding improves performance for extreme values and prevents
convergence exceptions.
</action>
<action dev="psteitz" type="update" issue="MATH-420">
Added toString() override to StatisticalSummaryValues.
</action>

View File

@ -164,16 +164,18 @@ public class NormalDistributionTest extends ContinuousDistributionAbstractTest
/**
* Check to make sure top-coding of extreme values works correctly.
* Verifies fix for JIRA MATH-167
* Verifies fixes for JIRA MATH-167, MATH-414
*/
public void testExtremeValues() throws Exception {
NormalDistribution distribution = (NormalDistribution) getDistribution();
distribution.setMean(0);
distribution.setStandardDeviation(1);
for (int i = 0; i < 100; i+=5) { // make sure no convergence exception
for (int i = 0; i < 100; i++) { // make sure no convergence exception
double lowerTail = distribution.cumulativeProbability(-i);
double upperTail = distribution.cumulativeProbability(i);
if (i < 10) { // make sure not top-coded
if (i < 9) { // make sure not top-coded
// For i = 10, due to bad tail precision in erf (MATH-364), 1 is returned
// TODO: once MATH-364 is resolved, replace 9 with 30
assertTrue(lowerTail > 0.0d);
assertTrue(upperTail < 1.0d);
}
@ -182,6 +184,12 @@ public class NormalDistributionTest extends ContinuousDistributionAbstractTest
assertTrue(upperTail > 0.99999);
}
}
assertEquals(distribution.cumulativeProbability(Double.MAX_VALUE), 1, 0);
assertEquals(distribution.cumulativeProbability(-Double.MAX_VALUE), 0, 0);
assertEquals(distribution.cumulativeProbability(Double.POSITIVE_INFINITY), 1, 0);
assertEquals(distribution.cumulativeProbability(Double.NEGATIVE_INFINITY), 0, 0);
}
public void testMath280() throws MathException {