我想找到
中a
系数的最小二乘解
z = (a0 + a1*x + a2*y + a3*x**2 + a4*x**2*y + a5*x**2*y**2 + a6*y**2 +
a7*x*y**2 + a8*x*y)
给定长度为20的数组x
,y
和z
。基本上我正在寻找等价于numpy.polyfit
的二进制多项式。
This question类似,但解决方案是通过MATLAB提供的。
答案 0 :(得分:13)
以下示例说明如何使用numpy.linalg.lstsq
执行此任务:
import numpy as np
x = np.linspace(0, 1, 20)
y = np.linspace(0, 1, 20)
X, Y = np.meshgrid(x, y, copy=False)
Z = X**2 + Y**2 + np.random.rand(*X.shape)*0.01
X = X.flatten()
Y = Y.flatten()
A = np.array([X*0+1, X, Y, X**2, X**2*Y, X**2*Y**2, Y**2, X*Y**2, X*Y]).T
B = Z.flatten()
coeff, r, rank, s = np.linalg.lstsq(A, B)
调整系数coeff
为:
array([ 0.00423365, 0.00224748, 0.00193344, 0.9982576 , -0.00594063,
0.00834339, 0.99803901, -0.00536561, 0.00286598])
请注意,coeff[3]
和coeff[6]
分别对应X**2
和Y**2
,并且它们接近1.
,因为示例数据是使用{{创建的1}}。
答案 1 :(得分:2)
很好的答案。只是添加代码以使用a系数的最小二乘解重建函数,
def poly2Dreco(X, Y, c):
return (c[0] + X*c[1] + Y*c[2] + X**2*c[3] + X**2*Y*c[4] + X**2*Y**2*c[5] +
Y**2*c[6] + X*Y**2*c[7] + X*Y*c[8])
答案 2 :(得分:2)
根据@Saullo和@Francisco的回答,我做了一个有用的功能:
conda install -c anaconda py-xgboost
结果拟合度可以通过以下方式可视化:
def polyfit2d(x, y, z, kx=3, ky=3, order=None):
'''
Two dimensional polynomial fitting by least squares.
Fits the functional form f(x,y) = z.
Notes
-----
Resultant fit can be plotted with:
np.polynomial.polynomial.polygrid2d(x, y, soln.reshape((kx+1, ky+1)))
Parameters
----------
x, y: array-like, 1d
x and y coordinates.
z: np.ndarray, 2d
Surface to fit.
kx, ky: int, default is 3
Polynomial order in x and y, respectively.
order: int or None, default is None
If None, all coefficients up to maxiumum kx, ky, ie. up to and including x^kx*y^ky, are considered.
If int, coefficients up to a maximum of kx+ky <= order are considered.
Returns
-------
Return paramters from np.linalg.lstsq.
soln: np.ndarray
Array of polynomial coefficients.
residuals: np.ndarray
rank: int
s: np.ndarray
'''
# grid coords
x, y = np.meshgrid(x, y)
# coefficient array, up to x^kx, y^ky
coeffs = np.ones((kx+1, ky+1))
# solve array
a = np.zeros((coeffs.size, x.size))
# for each coefficient produce array x^i, y^j
for index, (j, i) in enumerate(np.ndindex(coeffs.shape)):
# do not include powers greater than order
if order is not None and i + j > order:
arr = np.zeros_like(x)
else:
arr = coeffs[i, j] * x**i * y**j
a[index] = arr.flatten()
# do leastsq fitting and return leastsq result
return np.linalg.lstsq(a.T, np.ravel(z), rcond=None)
答案 3 :(得分:0)
请注意,如果 kx != ky
代码将失败,因为 j
和 i
索引在循环中反转。
您从 (j,i)
获得 enumerate(np.ndindex(coeffs.shape))
,但随后您将 coeffs
中的元素寻址为 coeffs[i,j]
。由于系数矩阵的形状由您要求使用的最大多项式阶数给出,如果 kx != ky
矩阵将是矩形的,并且您将超过其维度之一。