gpt4 book ai didi

python - 在类中使用 scipy.optimize.curve_fit

转载 作者:行者123 更新时间:2023-11-30 23:11:42 27 4
gpt4 key购买 nike

我有一个描述数学函数的类。该类需要能够对自身进行最小二乘拟合以适应传入的数据。即你可以调用这样的方法:

classinstance.Fit(x,y)

它会调整其内部变量以最适合数据。我正在尝试使用 scipy.optimize.curve_fit 为此,它需要我传递模型函数。问题在于模型函数位于类内部,需要访问类的变量和成员来计算数据。但是,curve_fit 无法调用第一个参数为 self 的函数。有没有办法让 curve_fit 使用类的方法作为模型函数?

这是显示问题的最小可执行代码片段:

import numpy as np
import matplotlib.pyplot as plt
from scipy.optimize import curve_fit

# This is a class which encapsulates a gaussian and fits itself to data.
class GaussianComponent():
# This is a formula string showing the exact code used to produce the gaussian. I
# It has to be printed for the user, and it can be used to compute values.
Formula = 'self.Amp*np.exp(-((x-self.Center)**2/(self.FWHM**2*np.sqrt(2))))'

# These parameters describe the gaussian.
Center = 0
Amp = 1
FWHM = 1

# HERE IS THE CONUNDRUM: IF I LEAVE SELF IN THE DECLARATION, CURVE_FIT
# CANNOT CALL IT SINCE IT REQUIRES THE WRONG NUMBER OF PARAMETERS.
# IF I REMOVE IT, FITFUNC CAN'T ACCESS THE CLASS VARIABLES.
def FitFunc(self, x, y, Center, Amp, FWHM):
eval('y - ' + self.Formula.replace('self.', ''))

# This uses curve_fit to adjust the gaussian parameters to best match the
# data passed in.
def Fit(self, x, y):
#FitFunc = lambda x, y, Center, Amp, FWHM: eval('y - ' + self.Formula.replace('self.', ''))
FitParams, FitCov = curve_fit(self.FitFunc, x, y, (self.Center, self.Amp, self.FWHM))
self.Center = FitParams[0]
self.Amp = FitParams[1]
self.FWHM = FitParams[2]

# Give back a vector which describes what this gaussian looks like.
def GetPlot(self, x):
y = eval(self.Formula)
return y

# Make a gausssian with default shape and position (height 1 at the origin, FWHM 1.
g = GaussianComponent()

# Make a space in which we can plot the gaussian.
x = np.linspace(-5,5,100)
y = g.GetPlot(x)

# Make some "experimental data" which is just the default shape, noisy, and
# moved up the y axis a tad so the best fit will be different.
ynoise = y + np.random.normal(loc=0.1, scale=0.1, size=len(x))

# Draw it
plt.plot(x,y, x,ynoise)
plt.show()

# Do the fit (but this doesn't work...)
g.Fit(x,y)

这会生成以下图表,然后崩溃,因为模型函数在尝试拟合时不正确。

enter image description here

提前致谢!

最佳答案

我花了一些时间查看你的代码,不幸的是迟到了 2 分钟。不管怎样,为了让事情变得更有趣,我对你的类(class)进行了一些编辑,这就是我的炮制内容:

import numpy as np
import matplotlib.pyplot as plt
from scipy.optimize import curve_fit

class GaussianComponent():

def __init__(self, func, params=None):
self.formula = func
self.params = params

def eval(self, x):
allowed_locals = {key: self.params[key] for key in self.params}
allowed_locals["x"] = x
allowed_globals = {"np":np}
return eval(self.formula, allowed_globals, allowed_locals)

def Fit(self, x, y):
FitParams, FitCov = curve_fit(self.eval, x, y, self.params)
self.fitparams = fitParams


# Make a gausssian with default shape and position (height 1 at the origin, FWHM 1.
g = GaussianComponent("Amp*np.exp(-((x-Center)**2/(FWHM**2*np.sqrt(2))))",
params={"Amp":1, "Center":0, "FWHM":1})

**SNIPPED FOR BREVITY**

我相信您也许会发现这是一个更令人满意的解决方案?

当前所有高斯参数都是类属性,这意味着如果您尝试使用不同的参数值创建类的第二个实例,您也将更改第一个类的值。通过将所有参数插入实例属性,您就可以摆脱它。这就是我们首先开设类(class)的原因。

您的 self 问题源于您在 Formula 中写入 self。现在您不必再这样做了。我认为这样更有意义,因为当您实例化类的对象时,您可以根据需要向声明的函数添加任意数量的参数。现在它甚至不必是高斯的(与以前不同)。

只需将所有参数放入字典中,就像 curve_fit 那样,然后忘记它们。

通过明确说明 eval 可以使用什么,您可以帮助确保任何作恶者更难破坏您的代码。不过,这仍然是可能的,它总是与 eval 一起使用。

祝你好运,询问您是否需要澄清一些事情。

关于python - 在类中使用 scipy.optimize.curve_fit,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30113580/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com