com.opengamma.strata.math.impl.integration.RungeKuttaIntegrator1D Maven / Gradle / Ivy
/*
* Copyright (C) 2009 - present by OpenGamma Inc. and the OpenGamma group of companies
*
* Please see distribution for license.
*/
package com.opengamma.strata.math.impl.integration;
import java.util.function.Function;
import com.opengamma.strata.collect.ArgChecker;
/**
* Adapted from the forth-order Runge-Kutta method for solving ODE. See here for the
* maths. It is a very robust integrator and should be used before trying more
* specialised methods.
*/
//CSOFF: JavadocMethod
public class RungeKuttaIntegrator1D extends Integrator1D {
private static final double DEF_TOL = 1e-10;
private static final double STEP_SIZE_LIMIT = 1e-50;
private static final int DEF_MIN_STEPS = 10;
private final double _absTol, _relTol;
private final int _minSteps;
/**
* Constructor from absolute and relative tolerance and minimal number of steps.
*
* The adaptable integration process stops when the difference between 2 steps is below the absolute tolerance
* plus the relative tolerance multiplied by the value.
*
* @param absTol the absolute tolerance
* @param relTol the relative tolerance
* @param minSteps the minimal number of steps
*/
public RungeKuttaIntegrator1D(double absTol, double relTol, int minSteps) {
if (absTol < 0.0 || Double.isNaN(absTol) || Double.isInfinite(absTol)) {
throw new IllegalArgumentException("Absolute Tolerance must be greater than zero");
}
if (relTol < 0.0 || Double.isNaN(relTol) || Double.isInfinite(relTol)) {
throw new IllegalArgumentException("Relative Tolerance must be greater than zero");
}
if (minSteps < 1) {
throw new IllegalArgumentException("Must have minimum of 1 step");
}
_absTol = absTol;
_relTol = relTol;
_minSteps = minSteps;
}
public RungeKuttaIntegrator1D(double tol, int minSteps) {
this(tol, tol, minSteps);
}
public RungeKuttaIntegrator1D(double atol, double rtol) {
this(atol, rtol, DEF_MIN_STEPS);
}
public RungeKuttaIntegrator1D(double tol) {
this(tol, tol, DEF_MIN_STEPS);
}
public RungeKuttaIntegrator1D(int minSteps) {
this(DEF_TOL, minSteps);
}
public RungeKuttaIntegrator1D() {
this(DEF_TOL, DEF_MIN_STEPS);
}
public double getRelativeTolerance() {
return _relTol;
}
@Override
public Double integrate(Function f, Double lower, Double upper) {
ArgChecker.notNull(lower, "lower");
ArgChecker.notNull(upper, "upper");
if (Double.isNaN(lower) || Double.isInfinite(lower) || Double.isInfinite(upper) || Double.isNaN(upper)) {
throw new IllegalArgumentException("lower or upper was NaN or Inf");
}
double h = (upper - lower) / _minSteps;
double f1, f2, f3, x;
x = lower;
f1 = f.apply(x);
if (Double.isNaN(f1) || Double.isInfinite(f1)) {
throw new IllegalArgumentException("function evaluation returned NaN or Inf");
}
double result = 0.0;
for (int i = 0; i < _minSteps; i++) {
f2 = f.apply(x + h / 2.0);
if (Double.isNaN(f2) || Double.isInfinite(f2)) {
throw new IllegalArgumentException("function evaluation returned NaN or Inf");
}
f3 = f.apply(x + h);
if (Double.isNaN(f3) || Double.isInfinite(f3)) {
throw new IllegalArgumentException("function evaluation returned NaN or Inf");
}
result += calculateRungeKuttaFourthOrder(f, x, h, f1, f2, f3);
f1 = f3;
x += h;
}
return result;
}
private double calculateRungeKuttaFourthOrder(
Function f,
double x,
double h,
double fl,
double fm,
double fu) {
// if (Double.isNaN(h) || Double.isInfinite(h) ||
// Double.isNaN(fl) || Double.isInfinite(fl) ||
// Double.isNaN(fm) || Double.isInfinite(fm) ||
// Double.isNaN(fu) || Double.isInfinite(fu)) {
// throw new OpenGammaRuntimeException("h was Inf or NaN");
// }
double f1 = f.apply(x + 0.25 * h);
if (Double.isNaN(f1) || Double.isInfinite(f1)) {
throw new IllegalStateException("f.evaluate returned NaN or Inf");
}
double f2 = f.apply(x + 0.75 * h);
if (Double.isNaN(f2) || Double.isInfinite(f2)) {
throw new IllegalStateException("f.evaluate returned NaN or Inf");
}
double ya = h * (fl + 4.0 * fm + fu) / 6.0;
double yb = h * (fl + 2.0 * fm + 4.0 * (f1 + f2) + fu) / 12.0;
double diff = Math.abs(ya - yb);
double abs = Math.max(Math.abs(ya), Math.abs(yb));
if (diff < _absTol + _relTol * abs) {
return yb + (yb - ya) / 15.0;
}
// can't keep halving the step size
if (h < STEP_SIZE_LIMIT) {
return yb + (yb - ya) / 15.0;
}
return calculateRungeKuttaFourthOrder(f, x, h / 2.0, fl, f1, fm) +
calculateRungeKuttaFourthOrder(f, x + h / 2.0, h / 2.0, fm, f2, fu);
}
}