MATH-855
The best point is sometimes not the last one evaluated. git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1381195 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
04b34cc1a5
commit
350f726ce5
|
@ -80,12 +80,13 @@ public class BrentOptimizer extends BaseAbstractUnivariateOptimizer {
|
||||||
if (abs <= 0) {
|
if (abs <= 0) {
|
||||||
throw new NotStrictlyPositiveException(abs);
|
throw new NotStrictlyPositiveException(abs);
|
||||||
}
|
}
|
||||||
|
|
||||||
relativeThreshold = rel;
|
relativeThreshold = rel;
|
||||||
absoluteThreshold = abs;
|
absoluteThreshold = abs;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The arguments are used implement the original stopping criterion
|
* The arguments are used for implementing the original stopping criterion
|
||||||
* of Brent's algorithm.
|
* of Brent's algorithm.
|
||||||
* {@code abs} and {@code rel} define a tolerance
|
* {@code abs} and {@code rel} define a tolerance
|
||||||
* {@code tol = rel |x| + abs}. {@code rel} should be no smaller than
|
* {@code tol = rel |x| + abs}. {@code rel} should be no smaller than
|
||||||
|
@ -226,7 +227,7 @@ public class BrentOptimizer extends BaseAbstractUnivariateOptimizer {
|
||||||
|
|
||||||
if (checker != null) {
|
if (checker != null) {
|
||||||
if (checker.converged(iter, previous, current)) {
|
if (checker.converged(iter, previous, current)) {
|
||||||
return current;
|
return best(current, previous, isMinim);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -263,9 +264,36 @@ public class BrentOptimizer extends BaseAbstractUnivariateOptimizer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else { // Default termination (Brent's criterion).
|
} else { // Default termination (Brent's criterion).
|
||||||
return current;
|
return best(current, previous, isMinim);
|
||||||
}
|
}
|
||||||
++iter;
|
++iter;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Selects the best of two points.
|
||||||
|
*
|
||||||
|
* @param a Point and value.
|
||||||
|
* @param b Point and value.
|
||||||
|
* @param isMinim {@code true} if the selected point must be the one with
|
||||||
|
* the lowest value.
|
||||||
|
* @return the best point, or {@code null} if {@code a} and {@code b} are
|
||||||
|
* both {@code null}.
|
||||||
|
*/
|
||||||
|
private UnivariatePointValuePair best(UnivariatePointValuePair a,
|
||||||
|
UnivariatePointValuePair b,
|
||||||
|
boolean isMinim) {
|
||||||
|
if (a == null) {
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
if (b == null) {
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isMinim) {
|
||||||
|
return a.getValue() < b.getValue() ? a : b;
|
||||||
|
} else {
|
||||||
|
return a.getValue() > b.getValue() ? a : b;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,8 @@ package org.apache.commons.math3.optimization.univariate;
|
||||||
import org.apache.commons.math3.analysis.QuinticFunction;
|
import org.apache.commons.math3.analysis.QuinticFunction;
|
||||||
import org.apache.commons.math3.analysis.UnivariateFunction;
|
import org.apache.commons.math3.analysis.UnivariateFunction;
|
||||||
import org.apache.commons.math3.analysis.function.Sin;
|
import org.apache.commons.math3.analysis.function.Sin;
|
||||||
|
import org.apache.commons.math3.analysis.function.StepFunction;
|
||||||
|
import org.apache.commons.math3.analysis.FunctionUtils;
|
||||||
import org.apache.commons.math3.exception.NumberIsTooLargeException;
|
import org.apache.commons.math3.exception.NumberIsTooLargeException;
|
||||||
import org.apache.commons.math3.exception.NumberIsTooSmallException;
|
import org.apache.commons.math3.exception.NumberIsTooSmallException;
|
||||||
import org.apache.commons.math3.exception.TooManyEvaluationsException;
|
import org.apache.commons.math3.exception.TooManyEvaluationsException;
|
||||||
|
@ -39,7 +41,7 @@ public final class BrentOptimizerTest {
|
||||||
public void testSinMin() {
|
public void testSinMin() {
|
||||||
UnivariateFunction f = new Sin();
|
UnivariateFunction f = new Sin();
|
||||||
UnivariateOptimizer optimizer = new BrentOptimizer(1e-10, 1e-14);
|
UnivariateOptimizer optimizer = new BrentOptimizer(1e-10, 1e-14);
|
||||||
Assert.assertEquals(3 * Math.PI / 2, optimizer.optimize(200, f, GoalType.MINIMIZE, 4, 5).getPoint(),1e-8);
|
Assert.assertEquals(3 * Math.PI / 2, optimizer.optimize(200, f, GoalType.MINIMIZE, 4, 5).getPoint(), 1e-8);
|
||||||
Assert.assertTrue(optimizer.getEvaluations() <= 50);
|
Assert.assertTrue(optimizer.getEvaluations() <= 50);
|
||||||
Assert.assertEquals(200, optimizer.getMaxEvaluations());
|
Assert.assertEquals(200, optimizer.getMaxEvaluations());
|
||||||
Assert.assertEquals(3 * Math.PI / 2, optimizer.optimize(200, f, GoalType.MINIMIZE, 1, 5).getPoint(), 1e-8);
|
Assert.assertEquals(3 * Math.PI / 2, optimizer.optimize(200, f, GoalType.MINIMIZE, 1, 5).getPoint(), 1e-8);
|
||||||
|
@ -181,4 +183,33 @@ public final class BrentOptimizerTest {
|
||||||
|
|
||||||
Assert.assertEquals(804.9355825, result, 1e-6);
|
Assert.assertEquals(804.9355825, result, 1e-6);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Contrived example showing that prior to the resolution of MATH-855,
|
||||||
|
* the algorithm, by always returning the last evaluated point, would
|
||||||
|
* sometimes not report the best point it had found.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testMath855() {
|
||||||
|
final double minSin = 3 * Math.PI / 2;
|
||||||
|
final double offset = 1e-8;
|
||||||
|
final double delta = 1e-7;
|
||||||
|
final UnivariateFunction f1 = new Sin();
|
||||||
|
final UnivariateFunction f2 = new StepFunction(new double[] { minSin, minSin + offset, minSin + 5 * offset },
|
||||||
|
new double[] { 0, -1, 0 });
|
||||||
|
final UnivariateFunction f = FunctionUtils.add(f1, f2);
|
||||||
|
final UnivariateOptimizer optimizer = new BrentOptimizer(1e-8, 1e-100);
|
||||||
|
final UnivariatePointValuePair result
|
||||||
|
= optimizer.optimize(200, f, GoalType.MINIMIZE, minSin - 6.789 * delta, minSin + 9.876 * delta);
|
||||||
|
final int numEval = optimizer.getEvaluations();
|
||||||
|
|
||||||
|
final double sol = result.getPoint();
|
||||||
|
final double expected = 4.712389027602411;
|
||||||
|
|
||||||
|
// System.out.println("min=" + (minSin + offset) + " f=" + f.value(minSin + offset));
|
||||||
|
// System.out.println("sol=" + sol + " f=" + f.value(sol));
|
||||||
|
// System.out.println("exp=" + expected + " f=" + f.value(expected));
|
||||||
|
|
||||||
|
Assert.assertTrue("Best point not reported", f.value(sol) <= f.value(expected));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue