Skip to main content

Flexible function solvers

Project description

license Version_status

What is flexsolve?

flexsolve presents a flexible set of function solvers by defining alternative tolerance conditions for accepting a solution. These solvers also implement methods like Wegstein and Aitken-Steffensen acceleration to reach solutions quicker.

Installation

Get the latest version of flexsolve from PyPI. If you have an installation of Python with pip, simple install it with:

$ pip install flexsolve

To get the git version, run:

$ git clone git://github.com/yoelcortes/flexsolve

Documentation

Flexsolve solvers can solve a variety of specifications:

  • Solve x where f(x) = x (iterative):

    • fixed_point: Simple fixed point iteration.

    • wegstein: Wegstein’s accelerated iteration method.

    • aitken: Aitken-Steffensen accelerated iteration method.

  • Solve x where f(x) = 0 and x0 < x < x1 (bounded):

    • bisection: Simple bisection method

    • false_position: Simple false position method.

    • IQ_interpolation: Quadratic interpolation solver (similar to scipy.optimize.brentq)

  • Solve x where f(x) = 0 (open):

    • secant: Simple secant method.

    • wegstein_secant: Secant method with Wegstein acceleration.

    • aitken_secant: Secant method with Aitken acceleration.

Parameters for each solver are pretty consitent and straight forward:

  • f: objective function in the form of f(x, *args).

  • x:

    • Iterative solvers: Root guess. Solver begins the iteration by evaluating f(x).

  • x0, x1:

    • Bounded solvers: Root bracket. Solution must lie within x0 and x1.

    • Open solvers: Initial and second guess. Second guess, ‘x1’, is optional.

  • xtol=1e-8: Solver stops when the root lies within xtol.

  • ytol=5e-8: Stop when the f(x) lies within ytol of the root.

  • args=(): Arguments to pass to f.

  • maxiter=50: Maximum number of iterations.

  • checkroot=True: Whether to raise a RuntimeError when root tolerance, ytol, is not satisfied.

Iterative solvers (which solve functions of the form f(x) = x) do not accept a ytol argument as xtol and ytol are actually mathematically equivalent.

Here are some exmples using flexsolve’s Profiler object to test and compare different solvers. In the graphs, the points are the solver iterations and the lines represent f(x). The lines and points are offset to make them more visible (so all the points are actually on the same curve!). The shaded area is just to help us relate the points to the curve (not an actual interval):

>>> import flexsolve as flx
>>> from scipy import optimize as opt
>>> x0, x1 = [-5, 5]
>>> f = lambda x: x**3 - 40 + 2*x
>>> p = flx.Profiler(f) # When called, it returns f(x) and saves the results.
>>> opt.brentq(p, x0, x1, xtol=1e-8)
3.225240462778411
>>> p.archive('[Scipy] Brent-Q') # Save/archive results with given name
>>> opt.brenth(p, x0, x1)
3.2252404627917794
>>> p.archive('[Scipy] Brent-H')
>>> flx.IQ_interpolation(p, x0, x1)
3.225240462796626
>>> p.archive('IQ-interpolation')
>>> flx.false_position(p, x0, x1)
3.225240462687035
>>> p.archive('False position')
>>> p.plot(r'$f(x) = 0 = x^3 + 2 \cdot x - 40$ where $-5 < x < 5$')
https://raw.githubusercontent.com/yoelcortes/flexsolve/master/docs/images/bounded_solvers_example.png
>>> p = flx.Profiler(f)
>>> x_guess = -5
>>> flx.aitken_secant(p, x_guess)
3.22524046279178
>>> p.archive('Aitken')
>>> flx.secant(p, x_guess)
3.2252404627918057
>>> p.archive('Secant')
>>> opt.newton(p, x_guess)
3.2252404627918065
>>> p.archive('[Scipy] Newton')
>>> p.plot(r'$f(x) = 0 = x^3 + 2 \cdot x - 40$')
https://raw.githubusercontent.com/yoelcortes/flexsolve/master/docs/images/general_solvers_example.png
>>> # Note that x = 40/x^2 - 2/x is the same
>>> # objective function as x**3 - 40 + 2*x = 0
>>> f = lambda x: 40/x**2 - 2/x
>>> p = flx.Profiler(f)
>>> x_guess = 5.
>>> flx.wegstein(p, x_guess)
3.2252404626726996
>>> p.archive('Wegstein')
>>> flx.aitken(p, x_guess)
3.2252404627250075
>>> p.archive('Aitken')
>>> p.plot(r'$f(x) = x = \frac{40}{x^2} - \frac{2}{x}$',
...        markbounds=False)
>>> # Fixed-point iteration is non-convergent for this equation,
>>> # so we do not include it here
https://raw.githubusercontent.com/yoelcortes/flexsolve/master/docs/images/fixed_point_solvers_example.png

If your project is need for speed, you can speed up calculations in flexsolve using the speed_up() method, which works by jit compiling computationally-heavy algorithms in flexsolve. The following example benchmarks flexsolve’s speed with and without compiling:

>>> import flexsolve as flx
>>> f = lambda x: x**3 - 40 + 2*x
>>> # Time solver without compiling
>>> %timeit flx.IQ_interpolation(f, -5, 5)
9.81 µs ± 131 ns per loop (mean ± std. dev. of 7 runs, 100000 loops each)
>>> flx.speed_up() # This is the only line we need to run to speed up flexsolve
>>> # First run is slower because it need to compile
>>> x = flx.IQ_interpolation(f, -5, 5)
>>> # Time solver after compiling
>>> %timeit flx.IQ_interpolation(f, -5, 5)
7.01 µs ± 88.4 ns per loop (mean ± std. dev. of 7 runs, 100000 loops each)

It is also possible to use compiled flexsolve solvers as part of jit-compiled code:

>>> from numba import njit
>>> import flexsolve as flx
>>> flx.speed_up() # Not necessary if previous example was run
>>> f = njit(lambda x: x**3 - 40 + 2*x) # Must be jit compiled to run in other compiled code
>>> # For demonstration purposes, the high level compiled function is a silly one liner
>>> solve_x = njit(lambda: flx.IQ_interpolation(f, -5., 5.))
>>> x = solve_x() # First run is slow because it needs to compile
>>> %timeit solve_x()
139 ns ± 2.08 ns per loop (mean ± std. dev. of 7 runs, 10000000 loops each)

The iterative methods for solving f(x) = x (e.g. fixed-point, Wegstain, Aitken) are capable of solving multi-dimensional problems. Simply make sure x is an array and f(x) returns an array with the same dimensions. In fact, the The Biorefinery Simulation and Techno-Economic Analysis Modules (BioSTEAM) uses flexsolve to solve many chemical engineering problems, including process recycle stream flow rates and vapor-liquid equili

Bug reports

To report bugs, please use the eqsolvers’s Bug Tracker at:

https://github.com/yoelcortes/flexsolve

License information

See LICENSE.txt for information on the terms & conditions for usage of this software, and a DISCLAIMER OF ALL WARRANTIES.

Although not required by the eqsolvers license, if it is convenient for you, please cite eqsolvers if used in your work. Please also consider contributing any changes you make back, and benefit the community.

Citation

To cite flexsolve in publications use:

Yoel Cortes-Pena (2019). flexsolve: Flexible function solvers. https://github.com/yoelcortes/flexsolve

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

flexsolve-0.3.14.tar.gz (13.7 kB view hashes)

Uploaded Source

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page