我有一些2D数据,特别是扫描的X光胶片。它们具有重叠点源曝光的测量值。数据示例:http://i.stack.imgur.com/oawlU.png
我想通过将2D高斯的总和拟合到数据来找到峰值位置。我已经尝试了其他一些方法并取得了一些成功,包括一个"星搜索"定位全局最大值的方法,拟合高斯并减去它。循环使用可以很好地找到所有峰值,但它不稳定且不太准确。我想使用星号搜索输出作为拟合的第一个猜测,但我遇到了麻烦,实施scipy.optimize.curve_fit
。
我创建了一个函数twoD_envelope
,它创建了从星搜索中找到的所有高斯的二维包络。这会产生此输出:http://i.stack.imgur.com/4KnpG.png
我的印象是我可以将其用作curve_fit
中的初始猜测,但我得到以下TypeError
:
TypeError: twoD_envelope() takes 4 positional arguments but 358802 were given
358802比数据大小多1,这是一个很大的线索,但我无法解决问题所在!我是一位物理学家,有着务实的态度。编码知识,所以非常感谢任何输入。
代码如下。
def twoD_envelope(npoints, xls, yls, pars):
envl = copy.copy(sq_image)
envl[:] = 0
for n in range(0,npoints):
height, cx, cy, width_x, width_y = pars[n]
FWHM = 0.5*(width_x+width_y)
g=makeGaussian(shape(sq_image)[0],fwhm=FWHM,center=[cx+xls[n],cy+yls[n]])
envl = envl + g
return envl.ravel()
# Create x and y indices
x = np.linspace(0, np.size(sq_image[0]), np.size(sq_image[0])+1)
y = np.linspace(0, np.size(sq_image[1]), np.size(sq_image[1])+1)
x, y = np.meshgrid(x, y)
coords = [x,y]
data = sq_image
initial_guess = twoD_envelope(9,xls,yls,pars)
pars_opt, pars_cov = opt.curve_fit(twoD_envelope, coords, data, p0=initial_guess)
(sq_image
是数据,ndarray
形状(599,599)
)
(pars, xls, yxl
=来自星搜索的高斯拟合参数列表)
(makeGaussian
是其他地方定义的函数)
答案 0 :(得分:1)
我认为您的错误消息是您传递给curve_fit
的函数的结果,该函数看起来不是正确的形式。这可能会导致data
数组被解释为vars
,从而导致您看到的TypeError
(无法运行代码很难分辨)。根据文档,传递给curve_fit的函数应该是
将自变量作为第一个参数,将参数作为单独的剩余参数。
您可能还需要在initial_guess之前使用星号将其作为元组传递,例如
pars_opt, pars_cov = opt.curve_fit(twoD_envelope, coords, data, p0=*initial_guess)
为了适应单个2D高斯,其中p0将是大约7个参数,有一个非常好的答案可能会有所帮助:Fitting a 2D Gaussian function using scipy.optimize.curve_fit - ValueError and minpack.error。
问题的一个可能解决方案可能是遍历您的2D sq_image并在本地使用单个2D高斯拟合,并使用星形搜索中的每个参数的初始参数...
编辑:适合高斯的代码。
import scipy.optimize as opt
import numpy as np
import pylab as plt
import matplotlib.cm as cm
import Image
def twoD_Gaussian((x, y), amplitude, xo, yo, sigma_x, sigma_y, theta):
xo = float(xo)
yo = float(yo)
a = (np.cos(theta)**2)/(2*sigma_x**2) + (np.sin(theta)**2)/(2*sigma_y**2)
b = -(np.sin(2*theta))/(4*sigma_x**2) + (np.sin(2*theta))/(4*sigma_y**2)
c = (np.sin(theta)**2)/(2*sigma_x**2) + (np.cos(theta)**2)/(2*sigma_y**2)
g = amplitude*np.exp( - (a*((x-xo)**2) + 2*b*(x-xo)*(y-yo)
+ c*((y-yo)**2)))
return g.ravel()
# Create x and y indices
I = Image.open('./30155885.png')
p = np.asarray(I).astype('float')
w,h = I.size
x, y = np.mgrid[0:h, 0:w]
#Use only one channel of image
p = p[:,:,0]
#Fit 2D Gaussian
initial_guess = (3,10,10,20,40,0)
popt, pcov = opt.curve_fit(twoD_Gaussian, (x, y), p.ravel(), p0=initial_guess)
data_fitted = twoD_Gaussian((x, y), *popt)
fig, ax = plt.subplots(1, 1)
cb = ax.imshow(p, cmap=plt.cm.jet, origin='bottom',
extent=(x.min(), x.max(), y.min(), y.max()))
ax.contour(x, y, data_fitted.reshape(x.shape[0], y.shape[1]), 8, colors='w')
plt.colorbar(cb)
plt.show()
图片30155885是,
请注意,只使用了一个图像数据通道(您应该使用sq_image中的数据替换数据数组)。这导致,