2016-07-15 75 views
1

T.A在学校向我展示了这个代码作为最小二乘拟合算法的一个例子。Cython/numpy vs纯numpy最小二乘拟合

import numpy as np 
#return the coefficients (a0,..aN) of the fit y=a0+a1*x+..an*x^n 
#with associated sigma dy 
#x,y,dy are all np.arrays with dtype= np.float64 
def fit_poly(x,y,dy,n): 
    V = np.asmatrix(np.diag(dy**2)) 
    M = [] 

    for k in range(n+1): 
     M.append(x**k) 
    M = np.asmatrix(M).T 
    theta = (M.T*V.I*M).I*M.T*V.I*np.asmatrix(y).T 
    cov_t = (M.T*V.I*M).I 

    return np.asarray(theta.T)[0], np.asarray(cov_t) 

我试图用cython优化他的代码。我得到这个代码

cimport numpy as np 
import numpy as np 
cimport cython 

@cython.boundscheck(False) 
@cython.wraparound(False) 
cpdef poly_c(np.ndarray[np.float64_t, ndim=1] x , 
    np.ndarray[np.float64_t, ndim=1] y np.ndarray[np.float64_t,ndim=1]dy , np.int n): 

    cdef np.ndarray[np.float64_t, ndim=2] V, M 

    V=np.asmatrix(np.diag(dy**2),dtype=np.float64) 
    M=np.asmatrix([x**k for k in range(n+1)],dtype=np.float64).T 

    return ((M.T*V.I*M).I*M.T*V.I*(np.asmatrix(y).T))[0],(M.T*V.I*M).I 

但运行时似乎是这两个程序是相同的,我没有使用“断言”,其中相同的,以确保输出。我错过了什么/做错了什么?

谢谢你的时间,希望你能帮助我。

PS:这是代码即时分析与(不知道我是否可以调用此配置,但W/E)

import numpy as np 
from polyC import poly_c 
from time import time 
from pancho_fit import fit_poly 
#pancho's the T.A,sup pancho 
x=np.arange(1,1000) 
x=np.asarray(x,dtype=np.float64) 
y=3*x+np.random.random(999) 
y=np.asarray(y,dtype=np.float64) 
dy=np.array([y.std() for i in range(1,1000)],dtype=np.float64) 
t0=time() 
a,b=poly_c(x,y,dy,4) 
#a,b=fit_poly(x,y,dy,4) 
print("time={}s".format(time()-t0)) 
+0

最好的办法是保存一些中间结果,您可以使用多次(例如'M.T * V.I')。当它们立即乘以另一个对速度和数字精度有利的矩阵时,通常还有避免矩阵求逆的方法。 – DavidW

回答

2

除了[x**k for k in range(n+1)]我没有看到任何迭代cython改善。大部分行动都在矩阵产品中。这些已经通过编译代码完成(np.dotndarrays)。

n只有4个,迭代次数不多。

但为什么迭代呢?

In [24]: x=np.arange(1,1000.) 
In [25]: M1=x[:,None]**np.arange(5) 
# np.matrix(M1) 

做同样的事情。

所以不,这看起来不像一个好的候选人 - 除非你准备把所有这些矩阵产品编写成可编译的细节。

我也跳过asmatrix东西,并使用常规的dot,@einsum,但这更多的是风格而不是速度。