scipy curve_fit 不喜欢数学模块

scipy curve_fit doesn't like math module

在尝试使用 scipy.optimize curve_fit 创建示例时,我发现 scipy 似乎与 Python 的 math 模块不兼容。虽然函数 f1 工作正常,但 f2 会抛出一条错误消息。

from scipy.optimize import curve_fit
from math import sin, pi, log, exp, floor, fabs, pow

x_axis = np.asarray([pi * i / 6 for i in range(-6, 7)])  
y_axis = np.asarray([sin(i) for i in x_axis])

def f1(x, m, n):
    return m * x + n

coeff1, mat = curve_fit(f1, x_axis, y_axis)    
print(coeff1)

def f2(x, m, n):
    return m * sin(x) + n 

coeff2, mat = curve_fit(f2, x_axis, y_axis)  
print(coeff2)

完整的追溯是

Traceback (most recent call last):
  File "/Documents/Programming/Eclipse/PythonDevFiles/so_test.py", line 49, in <module>
    coeff2, mat = curve_fit(f2, x_axis, y_axis)
  File "/usr/local/lib/python3.5/dist-packages/scipy/optimize/minpack.py", line 742, in curve_fit
    res = leastsq(func, p0, Dfun=jac, full_output=1, **kwargs)
  File "/usr/local/lib/python3.5/dist-packages/scipy/optimize/minpack.py", line 377, in leastsq
    shape, dtype = _check_func('leastsq', 'func', func, x0, args, n)
  File "/usr/local/lib/python3.5/dist-packages/scipy/optimize/minpack.py", line 26, in _check_func
    res = atleast_1d(thefunc(*((x0[:numinputs],) + args)))
  File "/usr/local/lib/python3.5/dist-packages/scipy/optimize/minpack.py", line 454, in func_wrapped
    return func(xdata, *params) - ydata
  File "/Documents/Programming/Eclipse/PythonDevFiles/so_test.py", line 47, in f2
    return m * sin(x) + n 
TypeError: only length-1 arrays can be converted to Python scalars

错误消息与列表和 numpy 数组一样作为输入出现。它影响所有 math 函数,我测试过(参见导入中的函数)并且必须与数学模块如何操作输入数据有关。这在 pow() 函数中最为明显 - 如果我不从 math 导入此函数,curve_fit 可以与 pow().

一起正常工作

显而易见的问题 - 为什么会发生这种情况以及 math 函数如何与 curve_fit 一起使用?

P.S.: 请不要讨论,那个不应该用线性拟合来拟合样本数据。这只是为了说明问题。

注意 numpy 数组、对数组的操作和对标量的操作!

Scipy optimize 假定输入(初始点)是一个一维数组,并且在其他情况下经常会出错(例如,一个列表变成一个数组,如果你假设在列表上工作,事情变得很糟糕;这类问题在 Whosebug 上很常见,并且调试不是那么容易通过眼睛进行;代码交互有帮助!)。

import numpy as np
import math

x = np.ones(1)

np.sin(x)
> array([0.84147098])

math.sin(x)
> 0.8414709848078965                     # this only works as numpy has dedicated support
                                         # as indicated by the error-msg below!
x = np.ones(2)

np.sin(x)
> array([0.84147098, 0.84147098])

math.sin(x)
> TypeError: only size-1 arrays can be converted to Python scalars

老实说:这是对 numpy 非常基本的理解的一部分,在使用 scipy 的一些敏感函数时应该理解。