Решение уравнения, содержащего абсолютное значение, с питоном 3 - PullRequest
0 голосов
/ 22 сентября 2019

Я пытаюсь решить следующее уравнение с питоном 3.6.3:

enter image description here

Я сделал

from sympy import *
x = Symbol('x', real=True)
solve(abs((abs(x**2-1)-x))-x)

но я получаю следующее сообщение:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "C:\Python36-32\lib\site-packages\sympy\solvers\solvers.py", line 1065, i
n solve
solution = _solve(f[0], *symbols, **flags)
  File "C:\Python36-32\lib\site-packages\sympy\solvers\solvers.py", line 1366, i
n _solve
candidates = _solve(piecewise_fold(expr), symbol, **flags)
  File "C:\Python36-32\lib\site-packages\sympy\solvers\solvers.py", line 1634, i
n _solve
raise NotImplementedError('\n'.join([msg, not_impl_msg % f]))
NotImplementedError: multiple generators [x, Abs(-x**2 + x + 1)]
No algorithms are implemented to solve equation -x + Abs(-x**2 + x + 1)

Но с python 2.7.14 и matlab я получаю ответы.Я что-то упустил?

1 Ответ

1 голос
/ 22 сентября 2019

Это дает [1, -1 + sqrt(2), 1 + sqrt(2)], если вы используете текущий мастер и вручную переписываете выражение как Piecewise.По-видимому, переписывание является неполным, когда оно выполнено самой solve:

>>> solve((abs((abs(x**2-1)-x))-x).rewrite(Piecewise))
[1, -1 + sqrt(2), 1 + sqrt(2)]

База кода SymPy может быть изменена с помощью следующего diff, чтобы исправить проблему:

diff --git a/sympy/solvers/solvers.py b/sympy/solvers/solvers.py
index 172d504..96bfa94 100644
--- a/sympy/solvers/solvers.py
+++ b/sympy/solvers/solvers.py
@@ -1020,8 +1020,13 @@ def _sympified_list(w):
         # Abs
         fi = fi.replace(Abs, lambda arg:
             separatevars(Abs(arg)) if arg.has(*symbols) else Abs(arg))
-        fi = fi.replace(Abs, lambda arg:
-            Abs(arg).rewrite(Piecewise) if arg.has(*symbols) else Abs(arg))
+        while True:
+            was = fi
+            fi = fi.replace(Abs, lambda arg:
+                (Abs(arg).rewrite(Piecewise) if arg.has(*symbols)
+                else Abs(arg)))
+            if was == fi:
+                break

         for e in fi.find(Abs):
             if e.has(*symbols):
...