Given N equispaced data-points \(F_i = F(t = i \Delta t)\), where \(i = 0, 1, ..., N-1\), the Prony method can be used to fit a sum of m decaying exponenitals: \[F(t) = \sum_{i=1}^{m} a_i e^{b_i t}. \] The 2m unknowns are \(a_i\) and \(b_i\).
Here is a python subprogram which implements the Prony method.
If you have arrays t and F, it can be called as:
a_est, b_est = prony(t, F, m)
In the Prony method, the number of modes in the exponential (m) is pre-specified. There are other methods, which are more general.
Here is a python subprogram which implements the Prony method.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def prony(t, F, m): | |
"""Input : real arrays t, F of the same size (ti, Fi) | |
: integer m - the number of modes in the exponential fit | |
Output : arrays a and b such that F(t) ~ sum ai exp(bi*t)""" | |
import numpy as np | |
import numpy.polynomial.polynomial as poly | |
# Solve LLS problem in step 1 | |
# Amat is (N-m)*m and bmat is N-m*1 | |
N = len(t) | |
Amat = np.zeros((N-m, m)) | |
bmat = F[m:N] | |
for jcol in range(m): | |
Amat[:, jcol] = F[m-jcol-1:N-1-jcol] | |
sol = np.linalg.lstsq(Amat, bmat) | |
d = sol[0] | |
# Solve the roots of the polynomial in step 2 | |
# first, form the polynomial coefficients | |
c = np.zeros(m+1) | |
c[m] = 1. | |
for i in range(1,m+1): | |
c[m-i] = -d[i-1] | |
u = poly.polyroots(c) | |
b_est = np.log(u)/(t[1] - t[0]) | |
# Set up LLS problem to find the "a"s in step 3 | |
Amat = np.zeros((N, m)) | |
bmat = F | |
for irow in range(N): | |
Amat[irow, :] = u**irow | |
sol = np.linalg.lstsq(Amat, bmat) | |
a_est = sol[0] | |
return a_est, b_est |
a_est, b_est = prony(t, F, m)