我在大型迭代中进行了以下两个矩阵代数计算。因此,我希望优化计算。
1:
F = np.matrix(np.zeros(shape=(n+1,1)))
F[0:n] = x - np.diag(np.array(theta)[0:n].flatten())*self.W*(theta[0:n]-self.m) + theta[0:n]*theta[n]
F[n] = np.sum(theta[0:n]) - 1; #Lagrange multiplier term
2:
J = np.matrix(np.zeros(shape=(n+1,n+1)))
#Now add the dF_lamba/d(theta_i) = 1 and dF_lamba/d(lambda) = 0
J[n,n] = 0
#The following is correct for the off diagonal elements
J[:n,:n] = -np.diag(np.array(theta)[0:n].flatten()) * self.W * np.diag(np.array(theta)[0:n].flatten())
#We now update for the on diagonal elements
J[:n,:n] = (J[:n,:n] - np.diag(np.diag(J[:n,:n])) +
np.diag(np.array(-np.multiply(np.diag(np.diag(self.W)),np.diag(np.array(theta)[0:n].flatten())) * self.W * (theta[0:n] - self.m) + theta[n]).flatten()))
#Finally adjust for the final columns
J[:n,n] = theta[:n]
J[n,:n] = 1
我不确定哪个numpy
调用计算量很大。是否有可能在Python中优化它以接近C速度,或者我是否必须用C本身编程?
测试功能1
import numpy as np
def _nonLinEq(m, W, x, theta):
#This outputs the nonlinear equations in theta
#resulting from a the partial log derivative of a multivariate
#normal prior with covariance matrix E, means m and a multiinomial
#likelihood with observations x.
#F = [F_1, ... F_n, F_lambda]' ouput values where F_i = F(theta_i)
n = len(m)
F = define_F(n)
F[0:n] = assign_values_to_F(x, theta, W, m, n)
F[n] = assign_lagrange_multiplier_term(theta, n) #Lagrange multiplier term
return F
def define_F(n):
return np.matrix(np.zeros(shape=(n+1,1)))
def diag_theta(theta, n):
return np.diag(np.array(theta)[0:n].flatten())
def multiply_terms(theta, W, m, n):
return diag_theta(theta, n)*W*(theta[0:n]-m)
def assign_values_to_F(x,theta,W,m,n):
return x - multiply_terms(theta, W, m, n) + theta[0:n]*theta[n]
def assign_lagrange_multiplier_term(theta, n):
return np.sum(theta[0:n]) - 1
def test_nonLinEq():
n = 100
temp = np.random.rand(n)
m = np.transpose(np.matrix(temp/np.sum(temp)))
W = np.matrix(np.diag(np.random.rand(n)))
x = np.transpose(np.matrix(np.floor(np.random.rand(n)*10)))
theta = np.transpose(np.matrix(np.append(np.ones(n)/n, -1)))
for i in range(1000):
_nonLinEq(m, W, x, theta)
2的测试功能
def _jacNLE(m, W, x, theta):
#This finds the Jacobian of our non-linear equations
#J = (J_ij) ouput values where F_ij = dF_i/d(theta_j)
n = len(m);
J = define_J(n)
#The following is correct for the off diagonal elements
diag_theta = convert_theta_to_diagonal(theta, n)
J[:n,:n] = input_off_diagonal_J(diag_theta, W)
#We now update for the on diagonal elements
J[:n,:n] = remove_J_diagonal(J, n) + new_diagonal(W, theta, m, diag_theta, n)
#Finally adjust for the final columns
J[:n,n] = theta[:n]
J[n,:n] = 1
return J
def define_J(n):
return np.matrix(np.zeros(shape=(n+1,n+1)))
def convert_theta_to_diagonal(theta, n):
return np.diag(np.array(theta)[0:n].ravel())
def input_off_diagonal_J(diag_theta, W):
return -diag_theta * W * diag_theta
def remove_J_diagonal(J, n):
return J[:n,:n] - np.diag(np.diag(J[:n,:n]))
def matrix_prod(W, diag_theta):
return -np.multiply(np.diag(np.diag(W)),diag_theta)
def new_diagonal(W, theta, m, diag_theta, n):
return np.diag(np.array(matrix_prod(W, diag_theta) * W * (theta[0:n] - m) + theta[n]).ravel())
def test_jacNLE():
n = 2
temp = np.random.rand(n)
m = np.transpose(np.matrix(temp/np.sum(temp)))
W = np.matrix(np.diag(np.random.rand(n)))
x = np.transpose(np.matrix(np.floor(np.random.rand(n)*10)))
theta = np.transpose(np.matrix(np.append(np.ones(n)/n, -1)))
for i in range(1000):
_jacNLE(m, W, x, theta)
答案 0 :(得分:1)
我最近想自己优化numpy代码。
首先我会尝试line_profiler package。它允许您测量每条线的时间消耗。起初使用起来有点棘手,但最后它是一个很好的工具,你可以肯定地发现瓶颈。
其他可能加快代码速度的工具是numba。如果您告诉某些类型信息,它会及时编译您的代码。
numpy中有特定的功能,我注意到它很慢。
首先:np.sum
它'奇怪但它甚至可能比内置sum
- 函数的pythons慢。我曾试图通过将你想要总结的数组乘以matrix(ones(N))
*或者在"中编写函数"来避免它。 numba
。我担心我不记得哪种解决方案更快。
我想我读过一次,np.matrix
比简单的包装np.asmatrix
慢。
如果您填写空数组,则可以使用np.empty
代替np.zeros
。
另一种方法是使用一些优化的BLAS,LAPACK东西再次在你的机器上编译numpy(直到我遇到了我从未听说过的那些问题......)它应该可以工作,但我失败了。
我也听说过ne gnumpy
- 使用CUDA
的套餐和你的GPU功能,但我不得不为此安装一些CUDA软件......
[编辑:] *我预定义onesN = matrix(ones(N))
所以每次我只想总结数组时都不需要构建它。这实际上是另一个提示:寻找常数并预定义并仔细阅读它们......
答案 1 :(得分:0)
我同意这些评论,你应该进行一些分析,但显然你可以做的一件事是分配
np.diag(np.array(theta)[0:n].flatten())
到一个局部变量并在你的计算中使用它,因为你多次这样做。如果你很大,这将是一个相当昂贵的计算,因为你是扁平化,复制到一个新的数组然后转而创建一个新的矩阵,该数组作为对角线。