added miscellaneous methods to get some feedback on the integration process

(step start, step size, number of evaluations ...)

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@919829 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Luc Maisonobe 2010-03-06 19:39:17 +00:00
parent fdd038890a
commit 9034808411
2 changed files with 94 additions and 12 deletions

View File

@ -25,6 +25,7 @@ import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import org.apache.commons.math.MathRuntimeException; import org.apache.commons.math.MathRuntimeException;
import org.apache.commons.math.MaxEvaluationsExceededException;
import org.apache.commons.math.ode.DerivativeException; import org.apache.commons.math.ode.DerivativeException;
import org.apache.commons.math.ode.FirstOrderDifferentialEquations; import org.apache.commons.math.ode.FirstOrderDifferentialEquations;
import org.apache.commons.math.ode.FirstOrderIntegrator; import org.apache.commons.math.ode.FirstOrderIntegrator;
@ -53,6 +54,12 @@ public class FirstOrderIntegratorWithJacobians {
/** Raw equations to integrate. */ /** Raw equations to integrate. */
private final ParameterizedODEWithJacobians ode; private final ParameterizedODEWithJacobians ode;
/** Maximal number of evaluations allowed. */
private int maxEvaluations;
/** Number of evaluations already performed. */
private int evaluations;
/** Build an enhanced integrator using internal differentiation to compute jacobians. /** Build an enhanced integrator using internal differentiation to compute jacobians.
* @param integrator underlying integrator to solve the compound problem * @param integrator underlying integrator to solve the compound problem
* @param ode original problem (f in the equation y' = f(t, y)) * @param ode original problem (f in the equation y' = f(t, y))
@ -74,6 +81,7 @@ public class FirstOrderIntegratorWithJacobians {
checkDimension(ode.getParametersDimension(), hP); checkDimension(ode.getParametersDimension(), hP);
this.integrator = integrator; this.integrator = integrator;
this.ode = new FiniteDifferencesWrapper(ode, p, hY, hP); this.ode = new FiniteDifferencesWrapper(ode, p, hY, hP);
setMaxEvaluations(-1);
} }
/** Build an enhanced integrator using ODE builtin jacobian computation features. /** Build an enhanced integrator using ODE builtin jacobian computation features.
@ -86,6 +94,7 @@ public class FirstOrderIntegratorWithJacobians {
final ParameterizedODEWithJacobians ode) { final ParameterizedODEWithJacobians ode) {
this.integrator = integrator; this.integrator = integrator;
this.ode = ode; this.ode = ode;
setMaxEvaluations(-1);
} }
/** Add a step handler to this integrator. /** Add a step handler to this integrator.
@ -204,7 +213,8 @@ public class FirstOrderIntegratorWithJacobians {
} }
// integrate the compound state variational equations // integrate the compound state variational equations
final double stopTime = integrator.integrate(new MappingWrapper(ode), t0, z, t, z); evaluations = 0;
final double stopTime = integrator.integrate(new MappingWrapper(), t0, z, t, z);
// dispatch the final compound state into the state and partial derivatives arrays // dispatch the final compound state into the state and partial derivatives arrays
System.arraycopy(z, 0, y, 0, n); System.arraycopy(z, 0, y, 0, n);
@ -219,6 +229,62 @@ public class FirstOrderIntegratorWithJacobians {
} }
/** Get the current value of the step start time t<sub>i</sub>.
* <p>This method can be called during integration (typically by
* the object implementing the {@link FirstOrderDifferentialEquations
* differential equations} problem) if the value of the current step that
* is attempted is needed.</p>
* <p>The result is undefined if the method is called outside of
* calls to <code>integrate</code>.</p>
* @return current value of the step start time t<sub>i</sub>
*/
public double getCurrentStepStart() {
return integrator.getCurrentStepStart();
}
/** Get the current signed value of the integration stepsize.
* <p>This method can be called during integration (typically by
* the object implementing the {@link FirstOrderDifferentialEquations
* differential equations} problem) if the signed value of the current stepsize
* that is tried is needed.</p>
* <p>The result is undefined if the method is called outside of
* calls to <code>integrate</code>.</p>
* @return current signed value of the stepsize
*/
public double getCurrentSignedStepsize() {
return integrator.getCurrentSignedStepsize();
}
/** Set the maximal number of differential equations function evaluations.
* <p>The purpose of this method is to avoid infinite loops which can occur
* for example when stringent error constraints are set or when lots of
* discrete events are triggered, thus leading to many rejected steps.</p>
* @param maxEvaluations maximal number of function evaluations (negative
* values are silently converted to maximal integer value, thus representing
* almost unlimited evaluations)
*/
public void setMaxEvaluations(int maxEvaluations) {
this.maxEvaluations = (maxEvaluations < 0) ? Integer.MAX_VALUE : maxEvaluations;
}
/** Get the maximal number of functions evaluations.
* @return maximal number of functions evaluations
*/
public int getMaxEvaluations() {
return maxEvaluations;
}
/** Get the number of evaluations of the differential equations function.
* <p>
* The number of evaluations corresponds to the last call to the
* <code>integrate</code> method. It is 0 if the method has not been called yet.
* </p>
* @return number of evaluations of the differential equations function
*/
public int getEvaluations() {
return evaluations;
}
/** Check array dimensions. /** Check array dimensions.
* @param expected expected dimension * @param expected expected dimension
* @param array (may be null if expected is 0) * @param array (may be null if expected is 0)
@ -234,10 +300,7 @@ public class FirstOrderIntegratorWithJacobians {
} }
/** Wrapper class used to map state and jacobians into compound state. */ /** Wrapper class used to map state and jacobians into compound state. */
private static class MappingWrapper implements FirstOrderDifferentialEquations { private class MappingWrapper implements FirstOrderDifferentialEquations {
/** Underlying ODE with jacobians. */
private final ParameterizedODEWithJacobians ode;
/** Current state. */ /** Current state. */
private final double[] y; private final double[] y;
@ -252,11 +315,8 @@ public class FirstOrderIntegratorWithJacobians {
private final double[][] dFdP; private final double[][] dFdP;
/** Simple constructor. /** Simple constructor.
* @param ode underlying ODE with jacobians
*/ */
public MappingWrapper(final ParameterizedODEWithJacobians ode) { public MappingWrapper() {
this.ode = ode;
final int n = ode.getDimension(); final int n = ode.getDimension();
final int k = ode.getParametersDimension(); final int k = ode.getParametersDimension();
@ -283,6 +343,9 @@ public class FirstOrderIntegratorWithJacobians {
// compute raw ODE and its jacobians: dy/dt, d[dy/dt]/dy0 and d[dy/dt]/dp // compute raw ODE and its jacobians: dy/dt, d[dy/dt]/dy0 and d[dy/dt]/dp
System.arraycopy(z, 0, y, 0, n); System.arraycopy(z, 0, y, 0, n);
if (++evaluations > maxEvaluations) {
throw new DerivativeException(new MaxEvaluationsExceededException(maxEvaluations));
}
ode.computeDerivatives(t, y, yDot); ode.computeDerivatives(t, y, yDot);
ode.computeJacobians(t, y, yDot, dFdY, dFdP); ode.computeJacobians(t, y, yDot, dFdY, dFdP);
@ -325,7 +388,7 @@ public class FirstOrderIntegratorWithJacobians {
} }
/** Wrapper class to compute jacobians by finite differences for ODE which do not compute them themselves. */ /** Wrapper class to compute jacobians by finite differences for ODE which do not compute them themselves. */
private static class FiniteDifferencesWrapper private class FiniteDifferencesWrapper
implements ParameterizedODEWithJacobians { implements ParameterizedODEWithJacobians {
/** Raw ODE without jacobians computation. */ /** Raw ODE without jacobians computation. */
@ -365,6 +428,8 @@ public class FirstOrderIntegratorWithJacobians {
/** {@inheritDoc} */ /** {@inheritDoc} */
public void computeDerivatives(double t, double[] y, double[] yDot) throws DerivativeException { public void computeDerivatives(double t, double[] y, double[] yDot) throws DerivativeException {
// this call to computeDerivatives has already been counted,
// we must not increment the counter again
ode.computeDerivatives(t, y, yDot); ode.computeDerivatives(t, y, yDot);
} }
@ -386,6 +451,11 @@ public class FirstOrderIntegratorWithJacobians {
final int n = ode.getDimension(); final int n = ode.getDimension();
final int k = ode.getParametersDimension(); final int k = ode.getParametersDimension();
evaluations += n + k;
if (evaluations > maxEvaluations) {
throw new DerivativeException(new MaxEvaluationsExceededException(maxEvaluations));
}
// compute df/dy where f is the ODE and y is the state array // compute df/dy where f is the ODE and y is the state array
for (int j = 0; j < n; ++j) { for (int j = 0; j < n; ++j) {
final double savedYj = y[j]; final double savedYj = y[j];

View File

@ -106,7 +106,12 @@ public class FirstOrderIntegratorWithJacobiansTest {
FirstOrderIntegratorWithJacobians extInt = FirstOrderIntegratorWithJacobians extInt =
new FirstOrderIntegratorWithJacobians(integ, brusselator, new double[] { b }, new FirstOrderIntegratorWithJacobians(integ, brusselator, new double[] { b },
new double[] { hY, hY }, new double[] { hP }); new double[] { hY, hY }, new double[] { hP });
extInt.setMaxEvaluations(5000);
extInt.integrate(0, z, new double[][] { { 0.0 }, { 1.0 } }, 20.0, z, dZdZ0, dZdP); extInt.integrate(0, z, new double[][] { { 0.0 }, { 1.0 } }, 20.0, z, dZdZ0, dZdP);
Assert.assertEquals(5000, extInt.getMaxEvaluations());
Assert.assertTrue(extInt.getEvaluations() > 2000);
Assert.assertTrue(extInt.getEvaluations() < 2500);
Assert.assertEquals(4 * integ.getEvaluations(), extInt.getEvaluations());
residualsP0.addValue(dZdP[0][0] - brusselator.dYdP0()); residualsP0.addValue(dZdP[0][0] - brusselator.dYdP0());
residualsP1.addValue(dZdP[1][0] - brusselator.dYdP1()); residualsP1.addValue(dZdP[1][0] - brusselator.dYdP1());
} }
@ -120,7 +125,7 @@ public class FirstOrderIntegratorWithJacobiansTest {
public void testAnalyticalDifferentiation() public void testAnalyticalDifferentiation()
throws IntegratorException, DerivativeException { throws IntegratorException, DerivativeException {
FirstOrderIntegrator integ = FirstOrderIntegrator integ =
new DormandPrince54Integrator(1.0e-8, 100.0, 1.0e-10, 1.0e-10); new DormandPrince54Integrator(1.0e-8, 100.0, 1.0e-4, 1.0e-4);
SummaryStatistics residualsP0 = new SummaryStatistics(); SummaryStatistics residualsP0 = new SummaryStatistics();
SummaryStatistics residualsP1 = new SummaryStatistics(); SummaryStatistics residualsP1 = new SummaryStatistics();
for (double b = 2.88; b < 3.08; b += 0.001) { for (double b = 2.88; b < 3.08; b += 0.001) {
@ -131,7 +136,12 @@ public class FirstOrderIntegratorWithJacobiansTest {
double[][] dZdP = new double[2][1]; double[][] dZdP = new double[2][1];
FirstOrderIntegratorWithJacobians extInt = FirstOrderIntegratorWithJacobians extInt =
new FirstOrderIntegratorWithJacobians(integ, brusselator); new FirstOrderIntegratorWithJacobians(integ, brusselator);
extInt.setMaxEvaluations(5000);
extInt.integrate(0, z, new double[][] { { 0.0 }, { 1.0 } }, 20.0, z, dZdZ0, dZdP); extInt.integrate(0, z, new double[][] { { 0.0 }, { 1.0 } }, 20.0, z, dZdZ0, dZdP);
Assert.assertEquals(5000, extInt.getMaxEvaluations());
Assert.assertTrue(extInt.getEvaluations() > 510);
Assert.assertTrue(extInt.getEvaluations() < 610);
Assert.assertEquals(integ.getEvaluations(), extInt.getEvaluations());
residualsP0.addValue(dZdP[0][0] - brusselator.dYdP0()); residualsP0.addValue(dZdP[0][0] - brusselator.dYdP0());
residualsP1.addValue(dZdP[1][0] - brusselator.dYdP1()); residualsP1.addValue(dZdP[1][0] - brusselator.dYdP1());
} }
@ -177,7 +187,7 @@ public class FirstOrderIntegratorWithJacobiansTest {
double[][] dydy0 = new double[2][2]; double[][] dydy0 = new double[2][2];
double[][] dydp = new double[2][3]; double[][] dydp = new double[2][3];
double t = 18 * Math.PI; double t = 18 * Math.PI;
FirstOrderIntegratorWithJacobians extInt = final FirstOrderIntegratorWithJacobians extInt =
new FirstOrderIntegratorWithJacobians(integ, circle); new FirstOrderIntegratorWithJacobians(integ, circle);
extInt.addStepHandler(new StepHandlerWithJacobians() { extInt.addStepHandler(new StepHandlerWithJacobians() {
@ -194,6 +204,8 @@ public class FirstOrderIntegratorWithJacobiansTest {
double[] y = interpolator.getInterpolatedY(); double[] y = interpolator.getInterpolatedY();
double[][] dydy0 = interpolator.getInterpolatedDyDy0(); double[][] dydy0 = interpolator.getInterpolatedDyDy0();
double[][] dydp = interpolator.getInterpolatedDyDp(); double[][] dydp = interpolator.getInterpolatedDyDp();
Assert.assertEquals(interpolator.getPreviousTime(), extInt.getCurrentStepStart(), 1.0e-10);
Assert.assertTrue(extInt.getCurrentSignedStepsize() < 0.5);
for (int i = 0; i < y.length; ++i) { for (int i = 0; i < y.length; ++i) {
Assert.assertEquals(circle.exactY(t)[i], y[i], 1.0e-10); Assert.assertEquals(circle.exactY(t)[i], y[i], 1.0e-10);
} }