3

以下是我从多元正态分布中绘制的 Cython 代码。我正在使用循环,因为每次我都有不同的密度。(conLSigma 是 Cholesky 因子)

这需要很多时间,因为我对每个循环都进行了逆分解和 Cholesky 分解。它比纯 python 代码快,但我想知道是否有任何方法可以提高速度。

from __future__ import division

import numpy as np 

cimport numpy as np 

ctypedef np.float64_t dtype_t

cimport cython
@cython.boundscheck(False)
@cython.wraparound(False)

def drawMetro(np.ndarray[dtype_t, ndim = 2] beta,
              np.ndarray[dtype_t, ndim = 3] H,
              np.ndarray[dtype_t, ndim = 2] Sigma,
              float s):

    cdef int ncons = betas.shape[0]
    cdef int nX = betas.shape[1]
    cdef int con

    cdef np.ndarray betas_cand = np.zeros([ncons, nX], dtype = np.float64)
    cdef np.ndarray conLSigma = np.zeros([nX, nX], dtype = np.float64)

    for con in xrange(ncons):
        conLSigma = np.linalg.cholesky(np.linalg.inv(H[con] + Sigma))
        betas_cand[con] = betas[con] + s * np.dot(conLSigma, np.random.standard_normal(size = nX))

    return(betas_cand)
4

2 回答 2

5

Cholesky 分解创建了一个下三角矩阵。这意味着np.dot不需要完成近一半的乘法运算。如果换行

betas_cand[con] = betas[con] + s * np.dot(conLSigma, np.random.standard_normal(size = nX))

进入

tmp = np.random.standard_normal(size = nX)
for i in xrange(nX):
    for j in xrange(i+1):
        betas_cand[con,i] += s * conLSigma[i,j] * tmp[j]

但是,您还需要更改

cdef np.ndarray betas_cand = np.zeros([ncons, nX], dtype = np.float64)

进入

cdef np.ndarray betas_cand = np.array(betas)

您当然可以使用切片进行乘法运算,但我不确定它是否会比我建议的方式更快。无论如何,希望你能明白这一点。我不认为你可以做很多其他事情来加快速度。

于 2010-11-21T06:36:11.620 回答
0

如何首先计算cholesky分解并在通过反向替换后反转下三角矩阵。这应该比 linalg.cholesky(linalg.inv(S)) 更快。

于 2014-06-09T15:03:23.737 回答