Numerical solver for finding function roots based on the improved algorithm proposed by Zhengqiu Zhang with new experimental counterpart points evaluation procedure.
AUTHOR: PIOTR JAZDZYK
LINKEDIN: https://www.linkedin.com/in/pjazdzyk
This is a Java implementation of a math algorithm for finding roots of continuous and single-variable functions, based on a modified version of the Brent-Dekker algorithm.
This solver uses a combination of Secant and Bisection numerical schemes and inverse quadratic interpolation to reduce calculation time as much as possible.
The algorithm code is based on the paper published by Zhengqiu Zhang in 2011. The new procedure has lower complexity and provides faster convergence than a classical
Brent-Dekker algorithm.
To mitigate the problematic issue by providing valid counterpart points <a,b> for which f(a) and f(b) will have opposite signs (meaning, that the root must be in the range between these points) the
automatic point evaluation algorithm was proposed, in which case solution will be found even for invalid points, but only if they
are close enough to the root and have a similar order of magnitude as the expected function root. However, this may not work
correctly for strong non-linearity or if the root is far away from any of the proposed points. The evaluation algorithm is an experimental
feature and will be subject to further development and improvement. Any suggestions on how to make it better are most welcome.
Not a thread-safe. This solver is not state-less and it is mutable. It is not allowed to use the same instance in concurrent applications. User must ensure that
each thread uses new instance of the solver.
Just copy Maven dependency provided below to your pom.xml file, and you are ready to go. For other package managers, check maven central repository: Brent-Dekker-Solver.
<dependency>
<groupId>com.synerset</groupId>
<artifactId>brent-dekker-solver</artifactId>
<version>2.0.0</version>
</dependency>
This code is my intellectual property. Please have respect for this. You can use it freely in any academic or non-commercial use if you properly include the source and author in your references or documentation. For commercial use, please contact me first.
- Iterative solver based on Brent-Dekker numerical scheme for solving any type of single variable equations.
- Improved convergence speed by implementing algorithm modifications proposed by Zhengqiu Zhang in his paper.
- Automatic counterpart evaluation algorithm to make solver less susceptible to invalid initial guess.
- Code review and comments, suggestions how to make this better.
- Convergence speedup and complexity reduction ideas.
- Counterpart points evaluation procedure improvement.
STEP 1: Brent-Dekker solver instance creation:
BrentSolver solver = new BrentSolver();
STEP 2: Provide equation as double function. Equation must be in form ie: a * x + b = 0:
- I. Linear function: 2x + 10 = 0
DoubleUnaryOperator linear = x -> 2 * x + 10;
- II. Quadratic function: 2x^2 +5x -3 = 0
DoubleUnaryOperator quadratic = x -> 2 * x * x + 5 * x -3;
- III. Complex function with nested argument under Logarithm
DoubleUnaryOperator logNested = x -> 93.3519196629417 - (-237300 * Math.log(0.001638 * x) / (1000 * Math.log(0.001638 * x) - 17269));
STEP 3: Provide counterpart points a and b, as a scope, between a solution is expected to be found. Function value for that points must result in opposite signs. If this fails, solver is equipped with the weak evaluation procedure, which will attempt to adjust your points to meet the criteria. If it does not succeed an exception is thrown. Let's try the first function. Default counterpart points are +50 / -50, lets try with these:
var resultLinear = solver.findRoot(linear);
// Outputs -5.0
solver outputs -5.0 what is the expected solution.
Now the more difficult part, the quadratic function. This function has two roots: -3.0, 0.5. Lets he how solver will
behave:
var resultQuadratic1stRoot = solver.findRoot(quadratic);
// Outputs -3.0
It is important to understand what happened above. The solver outputs the first root it finds and only one. If the found root is not the one you have expected - you may need to adjust the scope in which the desired solution can be found. Let's try to adjust counterpart points to get the other root:
solver.setCounterpartPoints(-1, 2);
// We can expect that second root is between 2 and 1
var resultQuadratic2ndRoot = solver.findRoot();
// Desired function is already set, so can simply invoke calculation method.
// Outputs 0.5, the second root
Last example is to test if solver will manage to find root for a complex coupled argument, for an example inside a logarithm:
solver.setFunction(logNested);
var resultLogNested = solver.findRoot();
If you run the code above, you will get an exception with information that NAN value was detected, which means that solution is not converged. The expected root for this function is a value of 80 000. The counterpart we last set are -1 and 2, which are several orders of magnitude smaller than any expected root. For such differences, automatic evaluation procedures will not manage to overcome it. Therefore, we need to change the solution range scope to a more appropriate threshold, for example, 20 000 - 200 000. Instead of invoking a separate counterpart setter, you can use an overloaded version of findRoot method:
var resultLogNested = solver.findRoot(logNested, 20000, 200000);
// Outputs: 79999.99999999991, with some small numerical error
Even if the provided counterpart points are invalid as shown above - the automatic evaluation algorithm will adjust them properly using linear extrapolation and few cycles of tries and error approach. This will work as long provided points are closed to the root and are of the similar order of magnitude. Is an extremely useful feature extending the usability of this solver. It may seem a bit of defensive approach, but my experience says that in many cases it will not be possible to automatically provide correct points each time. To see each step calculation you can turn on the diagnostic output.
solver.showDebugLogs(true);
resultLogNested = solver.findRoot(logNested, 100000, 200000);
// for these points f_a and f_b will not have an opposite signs
// despite invalid points, AB evaluation algorithms works fine
Despite the fact that the solution was not in the scope of provided counterpart points, the solver still managed to converge. You can observe the diagnostic output and see that the evaluation procedure algorithm was launched at the beginning and determined that second counterpart point needs to be changed to 74939.747 for the solver to work.
For single result, you can quickly use static method of() and get result immediately.
var functionRoot = BrentSolver.of(x -> 2 * x + 10, -10, 10);
// Outputs: -5.0
As it was demonstrated, this solver can handle almost any equation in a very few iterations steps and manages to provide a solution even if counterpart points are invalid (but still they need to be of the proper order of magnitude). The evaluation procedure is not bulletproof, I am looking forward to any improvement proposals from the community.
MIT LICENSE.
This work is licensed under the terms of the MIT License with the additional requirement that proper attribution be given
to the Piotr Jazdzyk as the original author in all derivative works and publications.
I have provided badges that you can include in your project to showcase your usage of our library:
Small shield with referenced most recent version tag:
[![Brent-Dekker-Solver](https://img.shields.io/github/v/release/pjazdzyk/brent-dekker-solver?label=Brent-Dekker%20solver&color=13ADF3&logo=)](https://github.com/pjazdzyk/brent-dekker-solver)
Tech shield with version tag for manual adjustment (you can indicate which version you actually use):
[![Brent-Dekker-Solver](https://img.shields.io/badge/Brent_Dekker%20solver-v2.0.0-13ADF3?style=for-the-badge&logo=)](https://github.com/pjazdzyk/brent-dekker-solver)
- [1] BRENT-DEKKER ITERATIVE SOLVER - MODIFIED ALGORITHM PROPOSED BY Zhengqiu Zhang / International Journal of Experimental Algorithms (IJEA), Volume (2) : Issue (1) : 2011
I extend my heartfelt gratitude to the Silesian University of Technology for imparting invaluable knowledge to me.
Thanks to Mathieu Soysal for his Maven central publisher.
Badges used in readme: Shields.io and Badges 4 README.md.