MATH-698
Bounds passed as arguments in "optimize" method. git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1194989 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
3ea635a01c
commit
5771d38200
|
@ -54,6 +54,13 @@ import org.apache.commons.math.util.MathArrays;
|
||||||
public class BOBYQAOptimizer
|
public class BOBYQAOptimizer
|
||||||
extends BaseAbstractScalarOptimizer<MultivariateRealFunction>
|
extends BaseAbstractScalarOptimizer<MultivariateRealFunction>
|
||||||
implements MultivariateRealOptimizer {
|
implements MultivariateRealOptimizer {
|
||||||
|
/** Minimum dimension of the problem: {@value} */
|
||||||
|
public static final int MINIMUM_PROBLEM_DIMENSION = 2;
|
||||||
|
/** Default value for {@link #initialTrustRegionRadius}: {@value} . */
|
||||||
|
public static final double DEFAULT_INITIAL_RADIUS = 10.0;
|
||||||
|
/** Default value for {@link #stoppingTrustRegionRadius}: {@value} . */
|
||||||
|
public static final double DEFAULT_STOPPING_RADIUS = 1E-8;
|
||||||
|
|
||||||
private static final double ZERO = 0d;
|
private static final double ZERO = 0d;
|
||||||
private static final double ONE = 1d;
|
private static final double ONE = 1d;
|
||||||
private static final double TWO = 2d;
|
private static final double TWO = 2d;
|
||||||
|
@ -67,13 +74,6 @@ public class BOBYQAOptimizer
|
||||||
private static final double ONE_OVER_TEN = ONE / 10;
|
private static final double ONE_OVER_TEN = ONE / 10;
|
||||||
private static final double ONE_OVER_A_THOUSAND = ONE / 1000;
|
private static final double ONE_OVER_A_THOUSAND = ONE / 1000;
|
||||||
|
|
||||||
/** Minimum dimension of the problem: {@value} */
|
|
||||||
public static final int MINIMUM_PROBLEM_DIMENSION = 2;
|
|
||||||
/** Default value for {@link #initialTrustRegionRadius}: {@value} . */
|
|
||||||
public static final double DEFAULT_INITIAL_RADIUS = 10.0;
|
|
||||||
/** Default value for {@link #stoppingTrustRegionRadius}: {@value} . */
|
|
||||||
public static final double DEFAULT_STOPPING_RADIUS = 1E-8;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* numberOfInterpolationPoints XXX
|
* numberOfInterpolationPoints XXX
|
||||||
*/
|
*/
|
||||||
|
@ -86,19 +86,6 @@ public class BOBYQAOptimizer
|
||||||
* stoppingTrustRegionRadius XXX
|
* stoppingTrustRegionRadius XXX
|
||||||
*/
|
*/
|
||||||
private final double stoppingTrustRegionRadius;
|
private final double stoppingTrustRegionRadius;
|
||||||
/**
|
|
||||||
* Lower bounds of the objective variables.
|
|
||||||
* {@code null} means no bounds.
|
|
||||||
* XXX Should probably be passed to the "optimize" method (overload not existing yet).
|
|
||||||
*/
|
|
||||||
private double[] lowerBound;
|
|
||||||
/**
|
|
||||||
* Upper bounds of the objective variables.
|
|
||||||
* {@code null} means no bounds.
|
|
||||||
* XXX Should probably be passed to the "optimize" method (overload not existing yet).
|
|
||||||
*/
|
|
||||||
private double[] upperBound;
|
|
||||||
|
|
||||||
/** Goal type (minimize or maximize). */
|
/** Goal type (minimize or maximize). */
|
||||||
private boolean isMinimize;
|
private boolean isMinimize;
|
||||||
/**
|
/**
|
||||||
|
@ -154,10 +141,10 @@ public class BOBYQAOptimizer
|
||||||
*/
|
*/
|
||||||
private ArrayRealVector gradientAtTrustRegionCenter;
|
private ArrayRealVector gradientAtTrustRegionCenter;
|
||||||
/**
|
/**
|
||||||
* Differences {@link #lowerBound} - {@link #originShift}.
|
* Differences {@link #getLowerBound()} - {@link #originShift}.
|
||||||
* All the components of every {@link #trustRegionCenterOffset} are going
|
* All the components of every {@link #trustRegionCenterOffset} are going
|
||||||
* to satisfy the bounds<br/>
|
* to satisfy the bounds<br/>
|
||||||
* {@link #lowerBound}<sub>i</sub> ≤
|
* {@link #getLowerBound() lowerBound}<sub>i</sub> ≤
|
||||||
* {@link #trustRegionCenterOffset}<sub>i</sub>,<br/>
|
* {@link #trustRegionCenterOffset}<sub>i</sub>,<br/>
|
||||||
* with appropriate equalities when {@link #trustRegionCenterOffset} is
|
* with appropriate equalities when {@link #trustRegionCenterOffset} is
|
||||||
* on a constraint boundary.
|
* on a constraint boundary.
|
||||||
|
@ -165,11 +152,11 @@ public class BOBYQAOptimizer
|
||||||
*/
|
*/
|
||||||
private ArrayRealVector lowerDifference;
|
private ArrayRealVector lowerDifference;
|
||||||
/**
|
/**
|
||||||
* Differences {@link #upperBound} - {@link #originShift}
|
* Differences {@link #getUpperBound()} - {@link #originShift}
|
||||||
* All the components of every {@link #trustRegionCenterOffset} are going
|
* All the components of every {@link #trustRegionCenterOffset} are going
|
||||||
* to satisfy the bounds<br/>
|
* to satisfy the bounds<br/>
|
||||||
* {@link #trustRegionCenterOffset}<sub>i</sub> ≤
|
* {@link #trustRegionCenterOffset}<sub>i</sub> ≤
|
||||||
* {@link #upperBound}<sub>i</sub>,<br/>
|
* {@link #getUpperBound() upperBound}<sub>i</sub>,<br/>
|
||||||
* with appropriate equalities when {@link #trustRegionCenterOffset} is
|
* with appropriate equalities when {@link #trustRegionCenterOffset} is
|
||||||
* on a constraint boundary.
|
* on a constraint boundary.
|
||||||
* XXX "su" in the original code.
|
* XXX "su" in the original code.
|
||||||
|
@ -223,23 +210,7 @@ public class BOBYQAOptimizer
|
||||||
* Choices that exceed {@code 2n+1} are not recommended.
|
* Choices that exceed {@code 2n+1} are not recommended.
|
||||||
*/
|
*/
|
||||||
public BOBYQAOptimizer(int numberOfInterpolationPoints) {
|
public BOBYQAOptimizer(int numberOfInterpolationPoints) {
|
||||||
this(numberOfInterpolationPoints, null, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param numberOfInterpolationPoints Number of interpolation conditions.
|
|
||||||
* For a problem of dimension {@code n}, its value must be in the interval
|
|
||||||
* {@code [n+2, (n+1)(n+2)/2]}.
|
|
||||||
* Choices that exceed {@code 2n+1} are not recommended.
|
|
||||||
* @param lowerBound Lower bounds (constraints) of the objective variables.
|
|
||||||
* @param upperBound Upperer bounds (constraints) of the objective variables.
|
|
||||||
*/
|
|
||||||
public BOBYQAOptimizer(int numberOfInterpolationPoints,
|
|
||||||
double[] lowerBound,
|
|
||||||
double[] upperBound) {
|
|
||||||
this(numberOfInterpolationPoints,
|
this(numberOfInterpolationPoints,
|
||||||
lowerBound,
|
|
||||||
upperBound,
|
|
||||||
DEFAULT_INITIAL_RADIUS,
|
DEFAULT_INITIAL_RADIUS,
|
||||||
DEFAULT_STOPPING_RADIUS);
|
DEFAULT_STOPPING_RADIUS);
|
||||||
}
|
}
|
||||||
|
@ -249,18 +220,12 @@ public class BOBYQAOptimizer
|
||||||
* For a problem of dimension {@code n}, its value must be in the interval
|
* For a problem of dimension {@code n}, its value must be in the interval
|
||||||
* {@code [n+2, (n+1)(n+2)/2]}.
|
* {@code [n+2, (n+1)(n+2)/2]}.
|
||||||
* Choices that exceed {@code 2n+1} are not recommended.
|
* Choices that exceed {@code 2n+1} are not recommended.
|
||||||
* @param lowerBound Lower bounds (constraints) of the objective variables.
|
|
||||||
* @param upperBound Upperer bounds (constraints) of the objective variables.
|
|
||||||
* @param initialTrustRegionRadius Initial trust region radius.
|
* @param initialTrustRegionRadius Initial trust region radius.
|
||||||
* @param stoppingTrustRegionRadius Stopping trust region radius.
|
* @param stoppingTrustRegionRadius Stopping trust region radius.
|
||||||
*/
|
*/
|
||||||
public BOBYQAOptimizer(int numberOfInterpolationPoints,
|
public BOBYQAOptimizer(int numberOfInterpolationPoints,
|
||||||
double[] lowerBound,
|
|
||||||
double[] upperBound,
|
|
||||||
double initialTrustRegionRadius,
|
double initialTrustRegionRadius,
|
||||||
double stoppingTrustRegionRadius) {
|
double stoppingTrustRegionRadius) {
|
||||||
this.lowerBound = lowerBound == null ? null : MathArrays.copyOf(lowerBound);
|
|
||||||
this.upperBound = upperBound == null ? null : MathArrays.copyOf(upperBound);
|
|
||||||
this.numberOfInterpolationPoints = numberOfInterpolationPoints;
|
this.numberOfInterpolationPoints = numberOfInterpolationPoints;
|
||||||
this.initialTrustRegionRadius = initialTrustRegionRadius;
|
this.initialTrustRegionRadius = initialTrustRegionRadius;
|
||||||
this.stoppingTrustRegionRadius = stoppingTrustRegionRadius;
|
this.stoppingTrustRegionRadius = stoppingTrustRegionRadius;
|
||||||
|
@ -269,13 +234,16 @@ public class BOBYQAOptimizer
|
||||||
/** {@inheritDoc} */
|
/** {@inheritDoc} */
|
||||||
@Override
|
@Override
|
||||||
protected RealPointValuePair doOptimize() {
|
protected RealPointValuePair doOptimize() {
|
||||||
|
final double[] lowerBound = getLowerBound();
|
||||||
|
final double[] upperBound = getUpperBound();
|
||||||
|
|
||||||
// Validity checks.
|
// Validity checks.
|
||||||
setup();
|
setup(lowerBound, upperBound);
|
||||||
|
|
||||||
isMinimize = (getGoalType() == GoalType.MINIMIZE);
|
isMinimize = (getGoalType() == GoalType.MINIMIZE);
|
||||||
currentBest = new ArrayRealVector(getStartPoint());
|
currentBest = new ArrayRealVector(getStartPoint());
|
||||||
|
|
||||||
final double value = bobyqa();
|
final double value = bobyqa(lowerBound, upperBound);
|
||||||
|
|
||||||
return new RealPointValuePair(currentBest.getDataRef(),
|
return new RealPointValuePair(currentBest.getDataRef(),
|
||||||
isMinimize ? value : -value);
|
isMinimize ? value : -value);
|
||||||
|
@ -311,9 +279,13 @@ public class BOBYQAOptimizer
|
||||||
* MAXFUN must be set to an upper bound on the number of calls of CALFUN.
|
* MAXFUN must be set to an upper bound on the number of calls of CALFUN.
|
||||||
* The array W will be used for working space. Its length must be at least
|
* The array W will be used for working space. Its length must be at least
|
||||||
* (NPT+5)*(NPT+N)+3*N*(N+5)/2.
|
* (NPT+5)*(NPT+N)+3*N*(N+5)/2.
|
||||||
* @return
|
*
|
||||||
|
* @param lowerBound Lower bounds.
|
||||||
|
* @param upperBound Upper bounds.
|
||||||
|
* @return the value of the objective at the optimum.
|
||||||
*/
|
*/
|
||||||
private double bobyqa() {
|
private double bobyqa(double[] lowerBound,
|
||||||
|
double[] upperBound) {
|
||||||
printMethod(); // XXX
|
printMethod(); // XXX
|
||||||
|
|
||||||
final int n = currentBest.getDimension();
|
final int n = currentBest.getDimension();
|
||||||
|
@ -359,7 +331,7 @@ public class BOBYQAOptimizer
|
||||||
|
|
||||||
// Make the call of BOBYQB.
|
// Make the call of BOBYQB.
|
||||||
|
|
||||||
return bobyqb();
|
return bobyqb(lowerBound, upperBound);
|
||||||
} // bobyqa
|
} // bobyqa
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------------------
|
||||||
|
@ -397,9 +369,12 @@ public class BOBYQAOptimizer
|
||||||
* W is a one-dimensional array that is used for working space. Its length
|
* W is a one-dimensional array that is used for working space. Its length
|
||||||
* must be at least 3*NDIM = 3*(NPT+N).
|
* must be at least 3*NDIM = 3*(NPT+N).
|
||||||
*
|
*
|
||||||
* @return
|
* @param lowerBound Lower bounds.
|
||||||
|
* @param upperBound Upper bounds.
|
||||||
|
* @return the value of the objective at the optimum.
|
||||||
*/
|
*/
|
||||||
private double bobyqb() {
|
private double bobyqb(double[] lowerBound,
|
||||||
|
double[] upperBound) {
|
||||||
printMethod(); // XXX
|
printMethod(); // XXX
|
||||||
|
|
||||||
final int n = currentBest.getDimension();
|
final int n = currentBest.getDimension();
|
||||||
|
@ -431,7 +406,7 @@ public class BOBYQAOptimizer
|
||||||
|
|
||||||
trustRegionCenterInterpolationPointIndex = 0;
|
trustRegionCenterInterpolationPointIndex = 0;
|
||||||
|
|
||||||
prelim();
|
prelim(lowerBound, upperBound);
|
||||||
double xoptsq = ZERO;
|
double xoptsq = ZERO;
|
||||||
for (int i = 0; i < n; i++) {
|
for (int i = 0; i < n; i++) {
|
||||||
trustRegionCenterOffset.setEntry(i, interpolationPoints.getEntry(trustRegionCenterInterpolationPointIndex, i));
|
trustRegionCenterOffset.setEntry(i, interpolationPoints.getEntry(trustRegionCenterInterpolationPointIndex, i));
|
||||||
|
@ -1604,8 +1579,11 @@ public class BOBYQAOptimizer
|
||||||
* KOPT will be such that the least calculated value of F so far is at
|
* KOPT will be such that the least calculated value of F so far is at
|
||||||
* the point XPT(KOPT,.)+XBASE in the space of the variables.
|
* the point XPT(KOPT,.)+XBASE in the space of the variables.
|
||||||
*
|
*
|
||||||
|
* @param lowerBound Lower bounds.
|
||||||
|
* @param upperBound Upper bounds.
|
||||||
*/
|
*/
|
||||||
private void prelim() {
|
private void prelim(double[] lowerBound,
|
||||||
|
double[] upperBound) {
|
||||||
printMethod(); // XXX
|
printMethod(); // XXX
|
||||||
|
|
||||||
final int n = currentBest.getDimension();
|
final int n = currentBest.getDimension();
|
||||||
|
@ -2397,10 +2375,13 @@ public class BOBYQAOptimizer
|
||||||
} // update
|
} // update
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Performs validity checks and adapt the {@link #lowerBound} and
|
* Performs validity checks.
|
||||||
* {@link #upperBound} array if no constraints were provided.
|
*
|
||||||
|
* @param lowerBound Lower bounds (constraints) of the objective variables.
|
||||||
|
* @param upperBound Upperer bounds (constraints) of the objective variables.
|
||||||
*/
|
*/
|
||||||
private void setup() {
|
private void setup(double[] lowerBound,
|
||||||
|
double[] upperBound) {
|
||||||
printMethod(); // XXX
|
printMethod(); // XXX
|
||||||
|
|
||||||
double[] init = getStartPoint();
|
double[] init = getStartPoint();
|
||||||
|
@ -2420,28 +2401,6 @@ public class BOBYQAOptimizer
|
||||||
nPointsInterval[1]);
|
nPointsInterval[1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check (and possibly adapt) bounds.
|
|
||||||
if (lowerBound == null) {
|
|
||||||
lowerBound = fillNewArray(dimension, Double.NEGATIVE_INFINITY);
|
|
||||||
} else if (lowerBound.length != init.length) {
|
|
||||||
throw new DimensionMismatchException(lowerBound.length, dimension);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (upperBound == null) {
|
|
||||||
upperBound = fillNewArray(dimension, Double.POSITIVE_INFINITY);
|
|
||||||
} else if (upperBound.length != init.length) {
|
|
||||||
throw new DimensionMismatchException(upperBound.length, dimension);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < dimension; i++) {
|
|
||||||
final double v = init[i];
|
|
||||||
final double lo = lowerBound[i];
|
|
||||||
final double hi = upperBound[i];
|
|
||||||
if (v < lo || v > hi) {
|
|
||||||
throw new OutOfRangeException(v, lo, hi);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize bound differences.
|
// Initialize bound differences.
|
||||||
boundDifference = new double[dimension];
|
boundDifference = new double[dimension];
|
||||||
|
|
||||||
|
|
|
@ -23,7 +23,7 @@ import org.apache.commons.math.analysis.MultivariateRealFunction;
|
||||||
import org.apache.commons.math.exception.DimensionMismatchException;
|
import org.apache.commons.math.exception.DimensionMismatchException;
|
||||||
import org.apache.commons.math.exception.TooManyEvaluationsException;
|
import org.apache.commons.math.exception.TooManyEvaluationsException;
|
||||||
import org.apache.commons.math.exception.NoDataException;
|
import org.apache.commons.math.exception.NoDataException;
|
||||||
import org.apache.commons.math.exception.OutOfRangeException;
|
import org.apache.commons.math.exception.NumberIsTooLargeException;
|
||||||
import org.apache.commons.math.exception.NumberIsTooSmallException;
|
import org.apache.commons.math.exception.NumberIsTooSmallException;
|
||||||
import org.apache.commons.math.optimization.GoalType;
|
import org.apache.commons.math.optimization.GoalType;
|
||||||
import org.apache.commons.math.optimization.MultivariateRealOptimizer;
|
import org.apache.commons.math.optimization.MultivariateRealOptimizer;
|
||||||
|
@ -39,7 +39,7 @@ public class BOBYQAOptimizerTest {
|
||||||
|
|
||||||
static final int DIM = 13;
|
static final int DIM = 13;
|
||||||
|
|
||||||
@Test(expected=OutOfRangeException.class)
|
@Test(expected=NumberIsTooLargeException.class)
|
||||||
public void testInitOutOfBounds() {
|
public void testInitOutOfBounds() {
|
||||||
double[] startPoint = point(DIM, 3);
|
double[] startPoint = point(DIM, 3);
|
||||||
double[][] boundaries = boundaries(DIM, -1, 2);
|
double[][] boundaries = boundaries(DIM, -1, 2);
|
||||||
|
@ -262,9 +262,8 @@ public class BOBYQAOptimizerTest {
|
||||||
// RealPointValuePair result = optim.optimize(100000, func, goal, startPoint);
|
// RealPointValuePair result = optim.optimize(100000, func, goal, startPoint);
|
||||||
final double[] lB = boundaries == null ? null : boundaries[0];
|
final double[] lB = boundaries == null ? null : boundaries[0];
|
||||||
final double[] uB = boundaries == null ? null : boundaries[1];
|
final double[] uB = boundaries == null ? null : boundaries[1];
|
||||||
MultivariateRealOptimizer optim =
|
MultivariateRealOptimizer optim = new BOBYQAOptimizer(2 * dim + 1);
|
||||||
new BOBYQAOptimizer(2 * dim + 1, lB, uB);
|
RealPointValuePair result = optim.optimize(maxEvaluations, func, goal, startPoint, lB, uB);
|
||||||
RealPointValuePair result = optim.optimize(maxEvaluations, func, goal, startPoint);
|
|
||||||
// System.out.println(func.getClass().getName() + " = "
|
// System.out.println(func.getClass().getName() + " = "
|
||||||
// + optim.getEvaluations() + " f(");
|
// + optim.getEvaluations() + " f(");
|
||||||
// for (double x: result.getPoint()) System.out.print(x + " ");
|
// for (double x: result.getPoint()) System.out.print(x + " ");
|
||||||
|
|
Loading…
Reference in New Issue