求解器¶
注意
对于专注于求解常见方程类型的初学者友好指南,请参阅 求解方程.
>>> from sympy import *
>>> x, y, z = symbols('x y z')
>>> init_printing(use_unicode=True)
关于方程的说明¶
回想一下本教程的 注意事项 部分,SymPy 中的符号方程不是由 =
或 ==
表示的,而是由 Eq
表示的。
>>> Eq(x, y)
x = y
然而,还有一个更简单的方法。在 SymPy 中,任何不在 Eq
中的表达式都会被求解函数自动认为等于 0。由于 \(a = b\) 当且仅当 \(a - b = 0\),这意味着您可以使用 x - y
代替 x == y
。例如
>>> solveset(Eq(x**2, 1), x)
{-1, 1}
>>> solveset(Eq(x**2 - 1, 0), x)
{-1, 1}
>>> solveset(x**2 - 1, x)
{-1, 1}
这在您要解的方程已经等于 0 时尤其有用。您可以使用 solveset(expr, x)
代替 solveset(Eq(expr, 0), x)
。
代数求解方程¶
求解代数方程的主要函数是 solveset
。 solveset
的语法是 solveset(equation, variable=None, domain=S.Complexes)
,其中 equations
可以是 Eq
实例或假定等于零的表达式。
请注意,还有一个名为 solve
的函数也可以用来求解方程。语法是 solve(equations, variables)
,但建议使用 solveset
代替。
当求解单个方程时,solveset
的输出是一个 FiniteSet
或 Interval
或 ImageSet
,包含解。
>>> solveset(x**2 - x, x)
{0, 1}
>>> solveset(x - x, x, domain=S.Reals)
ℝ
>>> solveset(sin(x) - 1, x, domain=S.Reals)
⎧ π │ ⎫
⎨2⋅n⋅π + ─ │ n ∊ ℤ⎬
⎩ 2 │ ⎭
如果没有解,则返回 EmptySet
,如果无法找到解,则返回 ConditionSet
。
>>> solveset(exp(x), x) # No solution exists
∅
>>> solveset(cos(x) - x, x) # Not able to find solution
{x │ x ∊ ℂ ∧ (-x + cos(x) = 0)}
在 solveset
模块中,线性方程组使用 linsolve
求解。将来我们能够直接从 solveset
中使用 linsolve
。以下是 linsolve
语法的示例。
方程列表形式
>>> linsolve([x + y + z - 1, x + y + 2*z - 3 ], (x, y, z)) {(-y - 1, y, 2)}
增广矩阵形式
>>> linsolve(Matrix(([1, 1, 1, 1], [1, 1, 2, 3])), (x, y, z)) {(-y - 1, y, 2)}
A*x = b 形式
>>> M = Matrix(((1, 1, 1, 1), (1, 1, 2, 3))) >>> system = A, b = M[:, :-1], M[:, -1] >>> linsolve(system, x, y, z) {(-y - 1, y, 2)}
注意
解的顺序对应于给定符号的顺序。
在 solveset
模块中,非线性方程组使用 nonlinsolve
求解。以下是 nonlinsolve
的示例。
只有实数解存在时
>>> a, b, c, d = symbols('a, b, c, d', real=True) >>> nonlinsolve([a**2 + a, a - b], [a, b]) {(-1, -1), (0, 0)} >>> nonlinsolve([x*y - 1, x - 2], x, y) {(2, 1/2)}
只有复数解存在时
>>> nonlinsolve([x**2 + 1, y**2 + 1], [x, y]) {(-ⅈ, -ⅈ), (-ⅈ, ⅈ), (ⅈ, -ⅈ), (ⅈ, ⅈ)}
实数解和复数解都存在时
>>> from sympy import sqrt >>> system = [x**2 - 2*y**2 -2, x*y - 2] >>> vars = [x, y] >>> nonlinsolve(system, vars) {(-2, -1), (2, 1), (-√2⋅ⅈ, √2⋅ⅈ), (√2⋅ⅈ, -√2⋅ⅈ)}
>>> system = [exp(x) - sin(y), 1/y - 3] >>> nonlinsolve(system, vars) {({2⋅n⋅ⅈ⋅π + log(sin(1/3)) │ n ∊ ℤ}, 1/3)}
当系统是正维系统(有无穷多解)时
>>> nonlinsolve([x*y, x*y - x], [x, y]) {(0, y)}
>>> system = [a**2 + a*c, a - b] >>> nonlinsolve(system, [a, b]) {(0, 0), (-c, -c)}
注意
解的顺序对应于给定符号的顺序。
2. 当前 nonlinsolve
不会以 LambertW
的形式返回解(如果存在 LambertW
形式的解)。
solve
可用于此类情况
>>> solve([x**2 - y**2/exp(x)], [x, y], dict=True)
⎡⎧ ____⎫ ⎧ ____⎫⎤
⎢⎨ ╱ x ⎬ ⎨ ╱ x ⎬⎥
⎣⎩y: -x⋅╲╱ ℯ ⎭, ⎩y: x⋅╲╱ ℯ ⎭⎦
>>> solve(x**2 - y**2/exp(x), x, dict=True)
⎡⎧ ⎛-y ⎞⎫ ⎧ ⎛y⎞⎫⎤
⎢⎨x: 2⋅W⎜───⎟⎬, ⎨x: 2⋅W⎜─⎟⎬⎥
⎣⎩ ⎝ 2 ⎠⎭ ⎩ ⎝2⎠⎭⎦
3. 当前 nonlinsolve
无法正确求解包含三角函数的方程组。
solve
可用于此类情况(但不会给出所有解)
>>> solve([sin(x + y), cos(x - y)], [x, y])
⎡⎛-3⋅π 3⋅π⎞ ⎛-π π⎞ ⎛π 3⋅π⎞ ⎛3⋅π π⎞⎤
⎢⎜─────, ───⎟, ⎜───, ─⎟, ⎜─, ───⎟, ⎜───, ─⎟⎥
⎣⎝ 4 4 ⎠ ⎝ 4 4⎠ ⎝4 4 ⎠ ⎝ 4 4⎠⎦
solveset
只报告每个解一次。要获取多项式的解,包括重数,请使用 roots
。
>>> solveset(x**3 - 6*x**2 + 9*x, x)
{0, 3}
>>> roots(x**3 - 6*x**2 + 9*x, x)
{0: 1, 3: 2}
roots
的输出 {0: 1, 3: 2}
表示 0
是重数为 1 的根,3
是重数为 2 的根。
注意
当前 solveset
无法求解以下类型的方程
可通过 LambertW(超越方程求解器)求解的方程。
solve
可用于此类情况
>>> solve(x*exp(x) - 1, x )
[W(1)]
求解微分方程¶
要求解微分方程,请使用 dsolve
。首先,通过将 cls=Function
传递给 symbols
函数来创建一个未定义函数。
>>> f, g = symbols('f g', cls=Function)
f
和 g
现在是未定义函数。我们可以调用 f(x)
,它将代表一个未知函数。
>>> f(x)
f(x)
f(x)
的导数是未计算的。
>>> f(x).diff(x)
d
──(f(x))
dx
(有关导数的更多信息,请参阅 导数 部分)。
要表示微分方程 \(f''(x) - 2f'(x) + f(x) = \sin(x)\),我们将使用
>>> diffeq = Eq(f(x).diff(x, x) - 2*f(x).diff(x) + f(x), sin(x))
>>> diffeq
2
d d
f(x) - 2⋅──(f(x)) + ───(f(x)) = sin(x)
dx 2
dx
要解 ODE,请将它和要解的函数传递给 dsolve
。
>>> dsolve(diffeq, f(x))
x cos(x)
f(x) = (C₁ + C₂⋅x)⋅ℯ + ──────
2
dsolve
返回 Eq
的实例。这是因为,通常情况下,微分方程的解无法显式地求解函数。
>>> dsolve(f(x).diff(x)*(1 - sin(f(x))) - 1, f(x))
x - f(x) - cos(f(x)) = C₁
来自 dsolve 的解中的任意常数是 C1
、C2
、C3
等等形式的符号。