python 在GPU上使用Sympy求解方程组

jk9hmnmh  于 2023-04-19  发布在  Python
关注(0)|答案(1)|浏览(217)

我目前正在使用solve()解决以下问题(高斯尾近似):

x, y = symbols('x y')
l_bound = np.mean(self.data[j]) - 3 * np.std(self.data[j])  # e.g. 4
u_bound = np.mean(self.data[j]) + 3 * np.std(self.data[j])  # e.g. 13
sol_dict = solve([y/(x-1) - 3 * y/sqrt(((x-1)**2)*(x-2))-l_bound, y/(x-1) + 3 * y/sqrt(((x-1)**2) * (x-2))-u_bound], [x, y], dict=True)

我试图找到一种方法来加快这个过程,因为它需要花费很长时间,而且我有相当多的数据矩阵行要处理(每个j是一行)。我在documentaries中找到的例子展示了如何计算函数-是否有可能在GPU上使用solve()?
到目前为止我所做的:我看过纪录片,找不到正确的表达方式。-因为我是新的主题,我可能错过了一些重要的部分(对不起,如果我的问题可能是微不足道的)。提前感谢任何指针在一个更合适的方向,感谢任何帮助。

pvcm50d1

pvcm50d11#

Sympy是一个完全基于Python的CAS,所以我不相信像求解器这样的复杂计算可以轻松地移植到GPU上。
然而,通过查看您的示例,似乎您正在尝试获得数值解。因此,solve不是正确的工具。nsolve(数值解)可能更适合这项工作。通常,您需要提供适当的初始猜测。例如:

l_bound = 4
u_bound = 13
expr1 = y/(x-1) - 3 * y/sqrt(((x-1)**2)*(x-2))-l_bound
expr2 = y/(x-1) + 3 * y/sqrt(((x-1)**2) * (x-2))-u_bound
nsolve([expr1, expr2], [x, y], [10, 10])
# out: (34.1111111111111, 281.444444444444)

你会注意到,对于这个特殊的例子,nsolvesolve快得多。然而,对于你的循环的每次迭代,nsolve将使用lambdify将符号方程组转换为数值函数组:这个操作是昂贵的。如果你的方程的形式没有随着迭代而改变,那么我们可以在开始时将符号方程系统转换为数值函数系统一次,然后在循环中使用更新的值重新使用它。例如:

from mpmath import findroot

l_bound, u_bound = symbols("l u")
expr1 = y/(x-1) - 3 * y/sqrt(((x-1)**2)*(x-2))-l_bound
expr2 = y/(x-1) + 3 * y/sqrt(((x-1)**2) * (x-2))-u_bound

system = Matrix([expr1, expr2]).T
J = system.jacobian([x, y])
# convert symbolic objects to numerical functions
system = lambdify([x, y, l_bound, u_bound], system.T, modules="mpmath")
J = lambdify([x, y, l_bound, u_bound], J, modules="mpmath")

l_bounds, u_bounds = [4], [13]
IG = [10, 10]
for l, u in zip(l_bounds, u_bounds):
    # provide new values of lower and upper bounds with each iteration
    sol = findroot(lambda x, y: system(x, y, l, u), IG, J=lambda x, y: J(x, y, l, u))
    # continue your task

备注:

  1. sympy的nsolve将完全执行我们在前面代码块中所做的操作:将方程组转换为符号矩阵,计算符号雅可比矩阵,将系统和雅可比矩阵转换为数值函数,并使用mpmath的findroot来找到解决方案。
    1.如果你不使用mpmath的findroot,而是使用Scipy的一些其他的求根算法,也许可以进一步提高性能。

相关问题