python - 通过 lmfit 模型在 python 中最小化拟合两个洛伦兹

标签 python curve-fitting minimize lmfit

也许有人可以帮助我。我花了几天时间,但我无法解决这个问题。提前致谢。

我想将 2 个洛伦兹人拟合到我的实验数据中。我将方程分解为两个洛伦兹 lorentz1lorentz2 函数的简单形式。然后我定义了另外两个函数 L1L2 只将一个常量 cnst 乘以它们。我有适合的所有 4 个参数:cnst1cnst2tau1tau2

我使用 lmfit:建模和最小化(可能都使用相同的方法)。

初始拟合参数的设置方式在视觉上更接近精细拟合。但是最小化使用 lmfit 会丢失(下面的第一张图片):


unconstrained parameters

使用这些参数:

params.add('cnst1', value=1e3 , min=1e2, max=1e5)
params.add('cnst2', value=3e5, min=1e2, max=1e6)
params.add('tau1', value=2e0, min=0, max=1e2)
params.add('tau2', value=5e-3, min=0, max=10)

但错误率很低:

cnst1:   117.459806 +/- 14.67188 (12.49%) (init= 1000)
cnst2:   413.959032 +/- 44.21042 (10.68%) (init= 300000)
tau1:    11.0343531 +/- 1.065570 (9.66%) (init= 2)
tau2:    1.55259664 +/- 0.125853 (8.11%) (init= 0.005)

另一方面,将参数包含在非常接近初始值的位置(强制与初始值一样):

enter image description here

使用参数:

#params.add('cnst1', value=1e3 , min=0.1e3, max=1e3)
#params.add('cnst2', value=3e5, min=1e3, max=1e6)
#params.add('tau1', value=2e0, min=0, max=2)
#params.add('tau2', value=5e-3, min=0, max=10)

拟合在视觉上更好,但错误值很大:

[[Variables]]
cnst1:   752.988629 +/- 221.3098 (29.39%) (init= 1000)
cnst2:   3.0159e+05 +/- 3.05e+07 (10113.40%) (init= 300000)
tau1:    1.99684317 +/- 0.600748 (30.08%) (init= 2)
tau2:    0.00497806 +/- 0.289651 (5818.56%) (init= 0.005)

这里是完整的代码:

import numpy as np
from lmfit import Model, minimize, Parameters, report_fit
import matplotlib.pyplot as plt

x = np.array([0.02988, 0.07057,0.19365,0.4137,0.91078,1.85075,3.44353,6.39428,\
        11.99302,24.37024,52.58804,121.71927,221.53799,358.27392,464.70405])

y = 1.0 / np.array([4.60362E-4,5.63559E-4,8.44538E-4,0.00138,0.00287,0.00657,0.01506,\
            0.03119,0.0584,0.09153,0.12538,0.19389,0.34391,0.68869,1.0])

def lorentz1(x, tau):
    L =  tau  / ( 1 + (x*tau)**2 )  
    return(L)

def lorentz2(x, tau):
    L =  tau**2  / ( 1 + (x*tau)**2 )  
    return(L)

def L1(x,cnst1,tau1):
    L1 =  cnst1 * lorentz1(x,tau1)
    return (L1)

def L2(x, cnst2, tau2):
    L2 =  cnst2 * lorentz2(x,tau2)
    return (L2)    

def L_min(params, x, y):
    cnst1 = params['cnst1'].value
    cnst2 = params['cnst2'].value
    tau1 = params['tau1'].value
    tau2 = params['tau2'].value

    L_total = L1(x, cnst1, tau1) + L2(x, cnst2, tau2)
    resids = L_total - y
    return resids

#params  = mod.make_params( cnst1=10e2, cnst2=3e5, tau1=2e0, tau2=0.5e-2)
params = Parameters()
#params.add('cnst1', value=1e3 , min=0.1e3, max=1e3)
#params.add('cnst2', value=3e5, min=1e3, max=1e6)
#params.add('tau1', value=2e0, min=0, max=2)
#params.add('tau2', value=5e-3, min=0, max=10)

params.add('cnst1', value=1e3 , min=1e2, max=1e5)
params.add('cnst2', value=3e5, min=1e2, max=1e6)
params.add('tau1', value=2e0, min=0, max=1e2)
params.add('tau2', value=5e-3, min=0, max=10)


#1-----Model--------------------
mod = Model(L1) + Model(L2)
result_mod = mod.fit(y, params, x=x)
print('---results from lmfit.Model----')
print(result_mod.fit_report())

#2---minimize-----------
result_min = minimize(L_min, params, args=(x,y))
final_min = y + result_min.residual
print('---results from lmfit.minimize----')
report_fit(params)

#-------Plot------
plt.close('all')
plt.loglog(x, y,'bo' , label='experimental data')
plt.loglog(x, result_mod.init_fit, 'k--', label='initial')
plt.loglog(x, result_mod.best_fit, 'r-', label='final')
plt.legend()
plt.show()

最佳答案

在搜索某些东西时,谷歌带来了我前段时间问过的自己的问题。现在我知道答案了,我在这里给出。我希望它能帮助别人。 :)

我会考虑 lmfit.minimize 函数。所以我所做的更改是绘制 lmfit.minimize 的结果。为了解决对数 y 尺度的问题(这是@mdurant 也提到的主要问题),我只是将残差除以它的 y 值(以某种方式标准化所有数据,以便在获取残差时具有可比性)。我将其命名为加权残差。

def L_min(params, x, y):
    ...
    ..
    .
    resids = L_total - y
    weighted_resids = resids/y
    return weighted_resids

因此结果由蓝线显示:

New fitting

完整代码:

import numpy as np
from lmfit import Model, minimize, Parameters, report_fit
import matplotlib.pyplot as plt

x = np.array([0.02988, 0.07057,0.19365,0.4137,0.91078,1.85075,3.44353,6.39428,\
        11.99302,24.37024,52.58804,121.71927,221.53799,358.27392,464.70405])

y = 1.0 / np.array([4.60362E-4,5.63559E-4,8.44538E-4,0.00138,0.00287,0.00657,0.01506,\
            0.03119,0.0584,0.09153,0.12538,0.19389,0.34391,0.68869,1.0])

def lorentz1(x, tau):
    L =  tau  / ( 1 + (x*tau)**2 )  
    return(L)

def lorentz2(x, tau):
    L =  tau**2  / ( 1 + (x*tau)**2 )  
    return(L)

def L1(x,cnst1,tau1):
    L1 =  cnst1 * lorentz1(x,tau1)
    return (L1)

def L2(x, cnst2, tau2):
    L2 =  cnst2 * lorentz2(x,tau2)
    return (L2)    

def L_min(params, x, y):
    cnst1 = params['cnst1'].value
    cnst2 = params['cnst2'].value
    tau1 = params['tau1'].value
    tau2 = params['tau2'].value

    L_total = L1(x, cnst1, tau1) + L2(x, cnst2, tau2)
    resids = L_total - y
    weighted_resids = resids/y
    return weighted_resids
#    return resids

#params  = mod.make_params( cnst1=10e2, cnst2=3e5, tau1=2e0, tau2=0.5e-2)
params = Parameters()
#params.add('cnst1', value=1e3 , min=0.1e3, max=1e3)
#params.add('cnst2', value=3e5, min=1e3, max=1e6)
#params.add('tau1', value=2e0, min=0, max=2)
#params.add('tau2', value=5e-3, min=0, max=10)

params.add('cnst1', value=1e3 , min=1e2, max=1e5)
params.add('cnst2', value=3e5, min=1e2, max=1e6)
params.add('tau1', value=2e0, min=0, max=1e2)
params.add('tau2', value=5e-3, min=0, max=10)


#1-----Model--------------------
mod = Model(L1) + Model(L2)
result_mod = mod.fit(y, params, x=x)
print('---results from lmfit.Model----')
print(result_mod.fit_report())

#2---minimize-----------
result_min = minimize(L_min, params, args=(x,y))
final_min = y + result_min.residual
print('---results from lmfit.minimize----')
report_fit(params)

#-------Plot------
plt.close('all')
plt.loglog(x, y,'bo' , label='experimental data')
plt.loglog(x, result_mod.init_fit, 'k--', label='initial')
plt.loglog(x, result_mod.best_fit, 'r-', label='lmfit.Model')

min_result = L1(x, params['cnst1'].value, params['tau1'].value) + \
            L2(x, params['cnst2'].value, params['tau2'].value)
plt.loglog(x, min_result, 'b-', label='lmfit.Minimize')
plt.legend()
plt.show()

拟合误差很好:

    cnst1:   832.592441 +/- 77.32939 (9.29%) (init= 1000)
    cnst2:   2.0836e+05 +/- 3.55e+04 (17.04%) (init= 300000)
    tau1:    1.64355457 +/- 0.221466 (13.47%) (init= 2)
    tau2:    0.00700899 +/- 0.000935 (13.34%) (init= 0.005)
[[Correlations]] (unreported correlations are <  0.100)
    C(tau1, tau2)                =  0.151 

关于python - 通过 lmfit 模型在 python 中最小化拟合两个洛伦兹,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26384926/

相关文章:

android - 尽量减少按下后退键时的 Activity

python - 使用 np.where 基于另一个维度设置一个 numpy 切片

python - 从 HDF5 文件列表创建一个 dask 数据框

python - 将 TensorFlow 与 Sage 结合使用

python - 为什么用try :?捕获异常时不能返回

python - 将闭合曲线拟合到一组点

python - python 中 3D 优化问题的并行化

python - Numba jit 与 scipy

r - 使用 nls 函数错误拟合

php - 最小化 HTML、PHP 或 CSS 文件对网页来说真的是一个很大的改进,还是没有什么大的区别?