fixed behavior of nextAfter(double, double) for infinity,
added nextAfter(float, float) and nextUp(float) JIRA: MATH-478 git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1061551 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
fef497d841
commit
9f30f38980
|
@ -599,6 +599,14 @@ public class FastMath {
|
||||||
return nextAfter(a, Double.POSITIVE_INFINITY);
|
return nextAfter(a, Double.POSITIVE_INFINITY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Compute next number towards positive infinity.
|
||||||
|
* @param a number to which neighbor should be computed
|
||||||
|
* @return neighbor of a towards positive infinity
|
||||||
|
*/
|
||||||
|
public static float nextUp(final float a) {
|
||||||
|
return nextAfter(a, Float.POSITIVE_INFINITY);
|
||||||
|
}
|
||||||
|
|
||||||
/** Returns a pseudo-random number between 0.0 and 1.0.
|
/** Returns a pseudo-random number between 0.0 and 1.0.
|
||||||
* @return a random number between 0.0 and 1.0
|
* @return a random number between 0.0 and 1.0
|
||||||
*/
|
*/
|
||||||
|
@ -3356,8 +3364,16 @@ public class FastMath {
|
||||||
public static double nextAfter(double d, double direction) {
|
public static double nextAfter(double d, double direction) {
|
||||||
|
|
||||||
// handling of some important special cases
|
// handling of some important special cases
|
||||||
if (Double.isNaN(d) || Double.isInfinite(d)) {
|
if (Double.isNaN(d)) {
|
||||||
return d;
|
return d;
|
||||||
|
} else if (Double.isInfinite(d)) {
|
||||||
|
if (d < direction) {
|
||||||
|
return -Double.MAX_VALUE;
|
||||||
|
} else if (direction < d) {
|
||||||
|
return Double.MAX_VALUE;
|
||||||
|
} else {
|
||||||
|
return d;
|
||||||
|
}
|
||||||
} else if (d == 0) {
|
} else if (d == 0) {
|
||||||
return (direction < 0) ? -Double.MIN_VALUE : Double.MIN_VALUE;
|
return (direction < 0) ? -Double.MIN_VALUE : Double.MIN_VALUE;
|
||||||
}
|
}
|
||||||
|
@ -3393,6 +3409,65 @@ public class FastMath {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the next machine representable number after a number, moving
|
||||||
|
* in the direction of another number.
|
||||||
|
* <p>
|
||||||
|
* If <code>direction</code> is greater than or equal to<code>d</code>,
|
||||||
|
* the smallest machine representable number strictly greater than
|
||||||
|
* <code>d</code> is returned; otherwise the largest representable number
|
||||||
|
* strictly less than <code>d</code> is returned.</p>
|
||||||
|
* <p>
|
||||||
|
* If <code>d</code> is NaN or Infinite, it is returned unchanged.</p>
|
||||||
|
*
|
||||||
|
* @param f base number
|
||||||
|
* @param direction (the only important thing is whether
|
||||||
|
* direction is greater or smaller than f)
|
||||||
|
* @return the next machine representable number in the specified direction
|
||||||
|
*/
|
||||||
|
public static float nextAfter(float f, float direction) {
|
||||||
|
|
||||||
|
// handling of some important special cases
|
||||||
|
if (Float.isNaN(f)) {
|
||||||
|
return f;
|
||||||
|
} else if (Float.isInfinite(f)) {
|
||||||
|
if (f < direction) {
|
||||||
|
return -Float.MAX_VALUE;
|
||||||
|
} else if (direction < f) {
|
||||||
|
return Float.MAX_VALUE;
|
||||||
|
} else {
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
} else if (f == 0f) {
|
||||||
|
return (direction < 0f) ? -Float.MIN_VALUE : Float.MIN_VALUE;
|
||||||
|
}
|
||||||
|
// special cases MAX_VALUE to infinity and MIN_VALUE to 0
|
||||||
|
// are handled just as normal numbers
|
||||||
|
|
||||||
|
// split the double in raw components
|
||||||
|
int bits = Float.floatToIntBits(f);
|
||||||
|
int sign = bits & 0x80000000;
|
||||||
|
int exponent = bits & 0x7f800000;
|
||||||
|
int mantissa = bits & 0x007fffff;
|
||||||
|
|
||||||
|
if (f * (direction - f) >= 0f) {
|
||||||
|
// we should increase the mantissa
|
||||||
|
if (mantissa == 0x000fffff) {
|
||||||
|
return Float.intBitsToFloat(sign | (exponent + 0x00800000));
|
||||||
|
} else {
|
||||||
|
return Float.intBitsToFloat(sign | exponent | (mantissa + 1));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// we should decrease the mantissa
|
||||||
|
if (mantissa == 0) {
|
||||||
|
return Float.intBitsToFloat(sign | (exponent - 0x00800000) | 0x007fffff);
|
||||||
|
} else {
|
||||||
|
return Float.intBitsToFloat(sign | exponent | (mantissa - 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/** Get the largest whole number smaller than x.
|
/** Get the largest whole number smaller than x.
|
||||||
* @param x number from which floor is requested
|
* @param x number from which floor is requested
|
||||||
* @return a double number f such that f is an integer f <= x < f + 1.0
|
* @return a double number f such that f is an integer f <= x < f + 1.0
|
||||||
|
|
|
@ -991,7 +991,7 @@ public class FastMathTest {
|
||||||
}
|
}
|
||||||
void check(Method mathMethod, double d, Object expected, Object actual, int entry){
|
void check(Method mathMethod, double d, Object expected, Object actual, int entry){
|
||||||
if (!expected.equals(actual)){
|
if (!expected.equals(actual)){
|
||||||
reportError(mathMethod.getName()+"(float "+d+") expected "+expected+" actual "+actual+ " entry "+entry);
|
reportError(mathMethod.getName()+"(double "+d+") expected "+expected+" actual "+actual+ " entry "+entry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue