001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017package org.apache.commons.math4.legacy.analysis.solvers;
018
019import org.apache.commons.math4.legacy.exception.NoBracketingException;
020import org.apache.commons.math4.legacy.exception.NumberIsTooLargeException;
021import org.apache.commons.math4.legacy.exception.TooManyEvaluationsException;
022import org.apache.commons.math4.core.jdkmath.JdkMath;
023
024/**
025 * This class implements the <a href="http://mathworld.wolfram.com/MullersMethod.html">
026 * Muller's Method</a> for root finding of real univariate functions. For
027 * reference, see <b>Elementary Numerical Analysis</b>, ISBN 0070124477,
028 * chapter 3.
029 * <p>
030 * Muller's method applies to both real and complex functions, but here we
031 * restrict ourselves to real functions.
032 * This class differs from {@link MullerSolver} in the way it avoids complex
033 * operations.</p><p>
034 * Except for the initial [min, max], it does not require bracketing
035 * condition, e.g. f(x0), f(x1), f(x2) can have the same sign. If a complex
036 * number arises in the computation, we simply use its modulus as a real
037 * approximation.</p>
038 * <p>
039 * Because the interval may not be bracketing, the bisection alternative is
040 * not applicable here. However in practice our treatment usually works
041 * well, especially near real zeroes where the imaginary part of the complex
042 * approximation is often negligible.</p>
043 * <p>
044 * The formulas here do not use divided differences directly.</p>
045 *
046 * @since 1.2
047 * @see MullerSolver
048 */
049public class MullerSolver2 extends AbstractUnivariateSolver {
050
051    /** Default absolute accuracy. */
052    private static final double DEFAULT_ABSOLUTE_ACCURACY = 1e-6;
053
054    /**
055     * Construct a solver with default accuracy (1e-6).
056     */
057    public MullerSolver2() {
058        this(DEFAULT_ABSOLUTE_ACCURACY);
059    }
060    /**
061     * Construct a solver.
062     *
063     * @param absoluteAccuracy Absolute accuracy.
064     */
065    public MullerSolver2(double absoluteAccuracy) {
066        super(absoluteAccuracy);
067    }
068    /**
069     * Construct a solver.
070     *
071     * @param relativeAccuracy Relative accuracy.
072     * @param absoluteAccuracy Absolute accuracy.
073     */
074    public MullerSolver2(double relativeAccuracy,
075                        double absoluteAccuracy) {
076        super(relativeAccuracy, absoluteAccuracy);
077    }
078
079    /**
080     * {@inheritDoc}
081     */
082    @Override
083    protected double doSolve()
084        throws TooManyEvaluationsException,
085               NumberIsTooLargeException,
086               NoBracketingException {
087        final double min = getMin();
088        final double max = getMax();
089
090        verifyInterval(min, max);
091
092        final double relativeAccuracy = getRelativeAccuracy();
093        final double absoluteAccuracy = getAbsoluteAccuracy();
094        final double functionValueAccuracy = getFunctionValueAccuracy();
095
096        // x2 is the last root approximation
097        // x is the new approximation and new x2 for next round
098        // x0 < x1 < x2 does not hold here
099
100        double x0 = min;
101        double y0 = computeObjectiveValue(x0);
102        if (JdkMath.abs(y0) < functionValueAccuracy) {
103            return x0;
104        }
105        double x1 = max;
106        double y1 = computeObjectiveValue(x1);
107        if (JdkMath.abs(y1) < functionValueAccuracy) {
108            return x1;
109        }
110
111        if(y0 * y1 > 0) {
112            throw new NoBracketingException(x0, x1, y0, y1);
113        }
114
115        double x2 = 0.5 * (x0 + x1);
116        double y2 = computeObjectiveValue(x2);
117
118        double oldx = Double.POSITIVE_INFINITY;
119        while (true) {
120            // quadratic interpolation through x0, x1, x2
121            final double q = (x2 - x1) / (x1 - x0);
122            final double a = q * (y2 - (1 + q) * y1 + q * y0);
123            final double b = (2 * q + 1) * y2 - (1 + q) * (1 + q) * y1 + q * q * y0;
124            final double c = (1 + q) * y2;
125            final double delta = b * b - 4 * a * c;
126            double x;
127            final double denominator;
128            if (delta >= 0.0) {
129                // choose a denominator larger in magnitude
130                double dplus = b + JdkMath.sqrt(delta);
131                double dminus = b - JdkMath.sqrt(delta);
132                denominator = JdkMath.abs(dplus) > JdkMath.abs(dminus) ? dplus : dminus;
133            } else {
134                // take the modulus of (B +/- JdkMath.sqrt(delta))
135                denominator = JdkMath.sqrt(b * b - delta);
136            }
137            if (denominator != 0) {
138                x = x2 - 2.0 * c * (x2 - x1) / denominator;
139                // perturb x if it exactly coincides with x1 or x2
140                // the equality tests here are intentional
141                while (x == x1 || x == x2) {
142                    x += absoluteAccuracy;
143                }
144            } else {
145                // extremely rare case, get a random number to skip it
146                x = min + JdkMath.random() * (max - min);
147                oldx = Double.POSITIVE_INFINITY;
148            }
149            final double y = computeObjectiveValue(x);
150
151            // check for convergence
152            final double tolerance = JdkMath.max(relativeAccuracy * JdkMath.abs(x), absoluteAccuracy);
153            if (JdkMath.abs(x - oldx) <= tolerance ||
154                JdkMath.abs(y) <= functionValueAccuracy) {
155                return x;
156            }
157
158            // prepare the next iteration
159            x0 = x1;
160            y0 = y1;
161            x1 = x2;
162            y1 = y2;
163            x2 = x;
164            y2 = y;
165            oldx = x;
166        }
167    }
168}