在Sympy中创建一个正式的线性函数

use*_*925 8 python sympy

我在Sympy中有一个表达(比如

-M - n + x(n)
Run Code Online (Sandbox Code Playgroud)

并且我想创建一个正式的线性函数,f表示,并将其应用于我的表达式,以便在简化后得到:

-f(M) - f(n) + f(x(n))
Run Code Online (Sandbox Code Playgroud)

是否有可能告诉同意线性的属性得到验证?

一种非常黑客的方法是将函数f应用于总和中的每个子表达式.例如,当给出类似我给出的第一个表达式时,简单地访问出现在总和中的术语会很好(这里它将是

[-M, -n , x(n)]
Run Code Online (Sandbox Code Playgroud)

然后在列表上映射f并将其求和以获得预期的值.

有没有一种简单的方法可以这样做,还是我必须通过表达式的句法树?

smi*_*chr 1

这有效:

>>> x,f = map(Function, 'xf'); n,M = symbols('n,M'); expr = -M - n + x(n)
>>> Add(*[f(a) for a in Add.make_args(expr)])
f(-M) + f(-n) + f(x(n))
Run Code Online (Sandbox Code Playgroud)

如果你有一个像这样的表达式f(n*(M + 1)),并将其展开,你会得到f(n*M + n)。你能告诉 SymPy 将函数应用于 的 args 的fargs 吗?是的:

>>> expr = f(n*(M + 1))
>>> expr.expand().replace(lambda x: x.func == f,
...         lambda x: Add(*[f(a) for a in Add.make_args(x.args[0])]))
f(n) + f(M*n)
Run Code Online (Sandbox Code Playgroud)

如果您调用这样的替换,linapp您可以将其用于您想要的任何功能:

def linapp(expr, *f):
    return expr.expand().replace(
      lambda x: x.func in f,
      lambda x: Add(*[x.func(a) for a in Add.make_args(x.args[0])]))

>>> print(linapp(cos(x+y) + sin(x + y), cos, sin))
sin(x) + sin(y) + cos(x) + cos(y)
Run Code Online (Sandbox Code Playgroud)

(并不是说这是一个真实的结果,只是说你可以做到。如果你用其他东西替换一个变量并且你想重新应用线性化,你可以:

>>> linapp(_.subs(y, z + 1), cos)
sin(x) + sin(z + 1) + cos(x) + cos(z) + cos(1)
Run Code Online (Sandbox Code Playgroud)