matplotlib 如何将曲线拟合为正弦函数

ojsjcaue  于 2023-10-24  发布在  其他
关注(0)|答案(2)|浏览(163)

我正在尝试使用正弦函数识别点阅读。您可以在图像上看到结果不令人满意。我无法调整curve_fit的各种参数以很好地拟合函数。我可以做些什么来改善我的结果?

  1. from scipy.optimize import curve_fit as cf
  2. import numpy as np
  3. from matplotlib import pyplot as plt
  4. xdata = np.array([0. , 0.02, 0.04, 0.06, 0.08, 0.1 , 0.12, 0.14, 0.16, 0.18, 0.2 ,
  5. 0.22, 0.24, 0.26, 0.28, 0.3 , 0.32, 0.34, 0.36, 0.38, 0.4 , 0.42,
  6. 0.44, 0.46, 0.48, 0.5 , 0.52, 0.54, 0.56, 0.58, 0.6 , 0.62, 0.64,
  7. 0.66, 0.68, 0.7 , 0.72, 0.74, 0.76, 0.78, 0.8 , 0.82, 0.84, 0.86,
  8. 0.88, 0.9 , 0.92, 0.94, 0.96, 0.98, 1. , 1.02, 1.04, 1.06, 1.08,
  9. 1.1 , 1.12])
  10. ydata = np.array([37.5, 36.4, 37.1, 37.3, 38.2, 38.4, 38.1, 36.7, 34.3, 32.2, 33.1,
  11. 31.8, 33.4, 35.7, 37.8, 38.3, 38.1, 37. , 34.9, 32.5, 31.6, 31.7,
  12. 33.5, 35.5, 37.8, 38.4, 38.3, 36.9, 34.8, 32.2, 33.9, 31.6, 33.3,
  13. 35.5, 37.6, 38.3, 38.2, 36.7, 34.8, 32.4, 31.5, 32.1, 33.3, 35.5,
  14. 37.2, 38.3, 38.3, 36.8, 34.5, 32.3, 31.6, 31.8, 33.3, 35.7, 37.8,
  15. 38.4, 38.2])
  16. def sin_fun(x,a,b,c,d):
  17. return a*np.sin(b*x+c)+d
  18. p_opt,p_cov=cf(sin_fun,xdata,ydata )
  19. print(p_opt)
  20. plt.plot(xdata,sin_fun(xdata,*p_opt))
  21. plt.plot(xdata,ydata, 'r')
  22. plt.show()

4bbkushb

4bbkushb1#

一般来说,@jared的答案是正确的。对于西内斯(或者,通常是任何周期函数),FFT是一种更可靠的方法,特别是如果它与二次峰值插值相结合:

  1. import matplotlib.pyplot as plt
  2. import numpy as np
  3. amplitude = np.array((
  4. 37.5, 36.4, 37.1, 37.3, 38.2, 38.4, 38.1, 36.7, 34.3, 32.2, 33.1,
  5. 31.8, 33.4, 35.7, 37.8, 38.3, 38.1, 37. , 34.9, 32.5, 31.6, 31.7,
  6. 33.5, 35.5, 37.8, 38.4, 38.3, 36.9, 34.8, 32.2, 33.9, 31.6, 33.3,
  7. 35.5, 37.6, 38.3, 38.2, 36.7, 34.8, 32.4, 31.5, 32.1, 33.3, 35.5,
  8. 37.2, 38.3, 38.3, 36.8, 34.5, 32.3, 31.6, 31.8, 33.3, 35.7, 37.8,
  9. 38.4, 38.2))
  10. timestep = 0.02
  11. time = np.arange(0, timestep*(len(amplitude) - 0.5), timestep)
  12. # first pass
  13. spectrum = np.fft.rfft(amplitude, norm='forward')
  14. spectral_amplitude = np.abs(spectrum)
  15. freqs = np.fft.rfftfreq(n=time.size, d=timestep)
  16. peak_index = 1 + spectral_amplitude[1:].argmax()
  17. # Instead of peak_freq = freqs[peak_index]
  18. # use parabolic peak interpolation from http://www.ericjacobsen.org/fe2/quadterp.txt
  19. a, b, c = spectrum[peak_index-1: peak_index+2]
  20. norm_frac = np.real((a - c)/(2*b - a - c))
  21. peak_freq_interpolated = freqs[peak_index]*(1 + norm_frac) - freqs[peak_index - 1]*norm_frac
  22. # second pass
  23. # shorten the timeseries length so that the peak lands in the next-left bin
  24. excerpt = int(np.round(time.size * freqs[peak_index-1] / peak_freq_interpolated))
  25. spectrum = np.fft.rfft(amplitude[-excerpt:], norm='forward')
  26. spectral_amplitude = np.abs(spectrum)
  27. freqs = np.fft.rfftfreq(n=excerpt, d=timestep)
  28. peak_index = 1 + spectral_amplitude[1:].argmax()
  29. peak_freq = freqs[peak_index]
  30. phase_correction = (1 - (len(time) - excerpt)*peak_freq*timestep)*2*np.pi
  31. peak_phase = np.angle(spectrum[peak_index]) + phase_correction
  32. peak_amplitude = 2*spectral_amplitude[peak_index]
  33. print(f'Amplitude: {peak_amplitude:.3f}')
  34. print(f'Frequency (Hz):', peak_freq)
  35. print(f'Phase (deg): {np.rad2deg(peak_phase):.2f}')
  36. reconstructed = peak_amplitude * np.cos(
  37. 2*np.pi*peak_freq*time + peak_phase
  38. ) + spectral_amplitude[0]
  39. ax_top: plt.Axes
  40. ax_bottom: plt.Axes
  41. fig, (ax_top, ax_bottom) = plt.subplots(nrows=2)
  42. ax_top.plot(time, amplitude, label='experiment')
  43. ax_top.plot(time, reconstructed, label='reconstructed')
  44. ax_top.set_xlabel('Interpolated time (s)')
  45. ax_top.set_ylabel('Amplitude')
  46. ax_top.legend(loc='lower right')
  47. ax_bottom.semilogy(freqs, spectral_amplitude)
  48. ax_bottom.set_xlabel('Frequency (Hz)')
  49. ax_bottom.set_ylabel('Amplitude')
  50. plt.show()
  1. Amplitude: 3.425
  2. Frequency (Hz): 5.0
  3. Phase (deg): 169.70

展开查看全部
4ktjp1zp

4ktjp1zp2#

你可以通过给函数一个好的初始猜测并为目标函数提供一个雅可比函数来改进你的结果。

  1. from scipy.optimize import curve_fit
  2. import numpy as np
  3. from matplotlib import pyplot as plt
  4. plt.close("all")
  5. xdata = np.array([0., 0.02, 0.04, 0.06, 0.08, 0.1, 0.12, 0.14, 0.16, 0.18, 0.2,
  6. 0.22, 0.24, 0.26, 0.28, 0.3, 0.32, 0.34, 0.36, 0.38, 0.4, 0.42,
  7. 0.44, 0.46, 0.48, 0.5, 0.52, 0.54, 0.56, 0.58, 0.6, 0.62, 0.64,
  8. 0.66, 0.68, 0.7, 0.72, 0.74, 0.76, 0.78, 0.8, 0.82, 0.84, 0.86,
  9. 0.88, 0.9, 0.92, 0.94, 0.96, 0.98, 1., 1.02, 1.04, 1.06, 1.08,
  10. 1.1, 1.12])
  11. ydata = np.array([37.5, 36.4, 37.1, 37.3, 38.2, 38.4, 38.1, 36.7, 34.3, 32.2, 33.1,
  12. 31.8, 33.4, 35.7, 37.8, 38.3, 38.1, 37., 34.9, 32.5, 31.6, 31.7,
  13. 33.5, 35.5, 37.8, 38.4, 38.3, 36.9, 34.8, 32.2, 33.9, 31.6, 33.3,
  14. 35.5, 37.6, 38.3, 38.2, 36.7, 34.8, 32.4, 31.5, 32.1, 33.3, 35.5,
  15. 37.2, 38.3, 38.3, 36.8, 34.5, 32.3, 31.6, 31.8, 33.3, 35.7, 37.8,
  16. 38.4, 38.2])
  17. def sin_fun(x, a, b, c, d):
  18. return a*np.sin(b*x+c)+d
  19. def jac(x, a, b, c, d):
  20. return np.stack([np.sin(b*x+c),
  21. a*x*np.cos(b*x+c),
  22. a*np.cos(b*x+c),
  23. np.ones_like(x)]).T
  24. guess = [3, 28, -0.5, 35]
  25. p_opt, p_cov = curve_fit(sin_fun, xdata, ydata, jac=jac, p0=guess)
  26. print(p_opt)
  27. x_fine = np.linspace(xdata.min(), xdata.max(), 1000)
  28. y_fine = sin_fun(x_fine, *p_opt)
  29. plt.plot(x_fine, y_fine)
  30. plt.plot(xdata, ydata, 'x')
  31. plt.show()

编辑:我在评论中被问到关于雅可比矩阵的问题,并给出了一个猜测。雅可比矩阵告诉曲线拟合函数,每个参数的变化将如何影响结果,它是通过对目标函数对每个参数的偏导数(按顺序)来计算的。对于猜测,我只是根据绘制的数据进行了目测。

展开查看全部

相关问题