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:
parent
fdd038890a
commit
9034808411
|
@ -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];
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue