An implementation of the Aberth-Ehrlich method for finding the zeros of a polynomial.
Aberth's method uses an electrostatics analogy to model the approximations as negative charges and the true zeros as positive charges. This enables finding all complex roots simultaneously, converging cubically (worst-case it converges linearly for zeros of multiplicity).
This crate is #![no_std]
& tries to have minimal dependencies. It uses
arrayvec to avoid allocations and
num-traits to be generic over floating
point types.
Add it to your project:
sh
cargo add aberth
Specify the coefficients of your polynomial in an array
rust
// 0 = -1 + 2x + 4x^4 + 11x^9
let polynomial = [-1.0, 2.0, 0.0, 0.0, 4.0, 0.0, 0.0, 0.0, 0.0, 11.0];
Call the aberth
method on your polynomial & provide an epsilon value.
```rust
use aberth::aberth;
let roots = aberth(&polynomial, EPSILON).unwrap(); // [ // Complex { real: 0.4293261, imaginary: 1.084202e-19 }, // Complex { real: 0.7263235, imaginary: 0.4555030 }, // Complex { real: 0.2067199, imaginary: 0.6750696 }, // Complex { real: -0.3448952, imaginary: 0.8425941 }, // Complex { real: -0.8028113, imaginary: 0.2296336 }, // Complex { real: -0.8028113, imaginary: -0.2296334 }, // Complex { real: -0.3448952, imaginary: -0.8425941 }, // Complex { real: 0.2067200, imaginary: -0.6750695 }, // Complex { real: 0.7263235, imaginary: -0.4555030 }, // ] ```
Note that the returned values are not sorted in any particular order.
The function returns an Err
if it fails to converge after 100 cycles. This
is excessive. Even polynomials of degree 20 will converge after <20 iterations.
Using a larger epsilon can usually avoid these errors.
The coefficient of the highest degree term should not be zero or you will get nonsense extra roots (probably at 0 + 0j).
This crate is dual-licensed under either of Apache license, Version 2.0 or the MIT license at your option.