All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.opengamma.strata.math.impl.rootfinding.BrentSingleRootFinder 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.rootfinding;

import java.util.function.Function;

import com.opengamma.strata.math.MathException;

/**
 * Root finder.
 */
public class BrentSingleRootFinder extends RealSingleRootFinder {

  private static final int MAX_ITER = 100;
  private static final double ZERO = 1e-16;
  private final double _accuracy;

  /**
   * Creates an instance.
   * Sets the accuracy to 10-15
   */
  public BrentSingleRootFinder() {
    this(1e-15);
  }

  /**
   * Creates an instance.
   * @param accuracy The accuracy of the root
   */
  public BrentSingleRootFinder(double accuracy) {
    _accuracy = accuracy;
  }

  //-------------------------------------------------------------------------
  @Override
  public Double getRoot(Function function, Double xLower, Double xUpper) {
    checkInputs(function, xLower, xUpper);
    if (xLower.equals(xUpper)) {
      return xLower;
    }
    double x1 = xLower;
    double x2 = xUpper;
    double x3 = xUpper;
    double delta = 0;
    double oldDelta = 0;
    double f1 = function.apply(x1);
    double f2 = function.apply(x2);
    double f3 = f2;
    double r1, r2, r3, r4, eps, xMid, min1, min2;
    for (int i = 0; i < MAX_ITER; i++) {
      if (f2 > 0 && f3 > 0 || f2 < 0 && f3 < 0) {
        x3 = x1;
        f3 = f1;
        delta = x2 - x1;
        oldDelta = delta;
      }
      if (Math.abs(f3) < Math.abs(f2)) {
        x1 = x2;
        x2 = x3;
        x3 = x1;
        f1 = f2;
        f2 = f3;
        f3 = f1;
      }
      eps = 2 * ZERO * Math.abs(x2) + 0.5 * _accuracy;
      xMid = (x3 - x2) / 2;
      if (Math.abs(xMid) <= eps) {
        return x2;
      }
      if (Math.abs(oldDelta) >= eps && Math.abs(f1) > Math.abs(f2)) {
        r4 = f2 / f1;
        if (Math.abs(x1 - x3) < ZERO) {
          r1 = 2 * xMid * r4;
          r2 = 1 - r4;
        } else {
          r2 = f1 / f3;
          r3 = f2 / f3;
          r1 = r4 * (2 * xMid * r2 * (r2 - r3) - (x2 - x1) * (r3 - 1));
          r2 = (r2 - 1) * (r3 - 1) * (r4 - 1);
        }
        if (r1 > 0) {
          r2 *= -1;
        }
        r1 = Math.abs(r1);
        min1 = 3 * xMid * r2 - Math.abs(eps * r2);
        min2 = Math.abs(oldDelta * r2);
        if (2 * r1 < (min1 < min2 ? min1 : min2)) {
          oldDelta = delta;
          delta = r1 / r2;
        } else {
          delta = xMid;
          oldDelta = delta;
        }
      } else {
        delta = xMid;
        oldDelta = delta;
      }
      x1 = x2;
      if (Math.abs(delta) > eps) {
        x2 += delta;
      } else {
        x2 += Math.copySign(eps, xMid);
      }
      f1 = function.apply(x1);
      f2 = function.apply(x2);
      f3 = function.apply(x3);
    }
    throw new MathException("Could not converge to root in " + MAX_ITER + " attempts");
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy