SymPy

SymPy - это пакет для символьных вычислений на питоне, подобный системе Mathematica. Он работает с выражениями, содержащими символы.

In [1]:
from sympy import *
init_printing()

Основными кирпичиками, из которых строятся выражения, являются символы. Символ имеет имя, которое используется при печати выражений. Объекты класса Symbol нужно создавать и присваивать переменным питона, чтобы их можно было использовать. В принципе, имя символа и имя переменной, которой мы присваиваем этот символ - две независимые вещи, и можно написать abc=Symbol('xyz'). Но тогда при вводе программы Вы будете использовать abc, а при печати результатов SymPy будет использовать xyz, что приведёт к ненужной путанице. Поэтому лучше, чтобы имя символа совпадало с именем переменной питона, которой он присваивается.

В языках, специально предназначенных для символьных вычислений, таких, как Mathematica, если Вы используете переменную, которой ничего не было присвоено, то она автоматически воспринимается как символ с тем же именем. Питон не был изначально предназначен для символьных вычислений. Если Вы используете переменную, которой ничего не было присвоено, Вы получите сообщение об ошибке. Объекты типа Symbol нужно создавать явно.

In [2]:
x=Symbol('x')
In [3]:
a=x**2-1
a
Out[3]:
$$x^{2} - 1$$
In [4]:
type(a)
Out[4]:
sympy.core.add.Add

Можно определить несколько символов одновременно. Строка разбивается на имена по пробелам.

In [5]:
y,z=symbols('y z')

Подставим вместо $x$ выражение $y+1$.

In [6]:
a.subs(x,y+1)
Out[6]:
$$\left(y + 1\right)^{2} - 1$$

Многочлены и рациональные функции

SymPy не раскрывает скобки автоматически. Для этого используется функция expand.

In [7]:
a=(x+y-z)**6
a
Out[7]:
$$\left(x + y - z\right)^{6}$$
In [8]:
a=expand(a)
a
Out[8]:
$$x^{6} + 6 x^{5} y - 6 x^{5} z + 15 x^{4} y^{2} - 30 x^{4} y z + 15 x^{4} z^{2} + 20 x^{3} y^{3} - 60 x^{3} y^{2} z + 60 x^{3} y z^{2} - 20 x^{3} z^{3} + 15 x^{2} y^{4} - 60 x^{2} y^{3} z + 90 x^{2} y^{2} z^{2} - 60 x^{2} y z^{3} + 15 x^{2} z^{4} + 6 x y^{5} - 30 x y^{4} z + 60 x y^{3} z^{2} - 60 x y^{2} z^{3} + 30 x y z^{4} - 6 x z^{5} + y^{6} - 6 y^{5} z + 15 y^{4} z^{2} - 20 y^{3} z^{3} + 15 y^{2} z^{4} - 6 y z^{5} + z^{6}$$

Степень многочлена $a$ по $x$.

In [9]:
degree(a,x)
Out[9]:
$$6$$

Соберём вместе члены с определёнными степенями $x$.

In [10]:
collect(a,x)
Out[10]:
$$x^{6} + x^{5} \left(6 y - 6 z\right) + x^{4} \left(15 y^{2} - 30 y z + 15 z^{2}\right) + x^{3} \left(20 y^{3} - 60 y^{2} z + 60 y z^{2} - 20 z^{3}\right) + x^{2} \left(15 y^{4} - 60 y^{3} z + 90 y^{2} z^{2} - 60 y z^{3} + 15 z^{4}\right) + x \left(6 y^{5} - 30 y^{4} z + 60 y^{3} z^{2} - 60 y^{2} z^{3} + 30 y z^{4} - 6 z^{5}\right) + y^{6} - 6 y^{5} z + 15 y^{4} z^{2} - 20 y^{3} z^{3} + 15 y^{2} z^{4} - 6 y z^{5} + z^{6}$$

Многочлен с целыми коэффициентами можно записать в виде произведения таких многочленов (причём каждый сомножитель уже невозможно расфакторизовать дальше, оставаясь в рамках многочленов с целыми коэффициентами). Существуют эффективные алгоритмы для решения этой задачи.

In [11]:
a=factor(a)
a
Out[11]:
$$\left(x + y - z\right)^{6}$$

SymPy не сокращает отношения многочленов на их наибольший общий делитель автоматически. Для этого используется функция cancel.

In [12]:
a=(x**3-y**3)/(x**2-y**2)
a
Out[12]:
$$\frac{x^{3} - y^{3}}{x^{2} - y^{2}}$$
In [13]:
cancel(a)
Out[13]:
$$\frac{x^{2} + x y + y^{2}}{x + y}$$

SymPy не приводит суммы рациональных выражений к общему знаменателю автоматически. Для этого используется функция together.

In [14]:
a=y/(x-y)+x/(x+y)
a
Out[14]:
$$\frac{x}{x + y} + \frac{y}{x - y}$$
In [15]:
together(a)
Out[15]:
$$\frac{x \left(x - y\right) + y \left(x + y\right)}{\left(x - y\right) \left(x + y\right)}$$

Функция simplify пытается переписать выражение в наиболее простом виде. Это понятие не имеет чёткого определения (в разных ситуациях наиболее простыми могут считаться разные формы выражения), и не существует алгоритма такого упрощения. Функция symplify работает эвристически, и невозможно заранее предугадать, какие упрощения она попытается сделать. Поэтому её удобно использовать в интерактивных сессиях, чтобы посмотреть, удастся ли ей записать выражение в каком-нибудь разумном виде, но нежелательно использовать в программах. В них лучше применять более специализированные функции, которые выполняют одно определённое преобразование выражения.

In [16]:
simplify(a)
Out[16]:
$$\frac{x^{2} + y^{2}}{x^{2} - y^{2}}$$

Разложение на элементарные дроби по отношению к $x$ и $y$.

In [17]:
apart(a,x)
Out[17]:
$$- \frac{y}{x + y} + \frac{y}{x - y} + 1$$
In [18]:
apart(a,y)
Out[18]:
$$\frac{x}{x + y} + \frac{x}{x - y} - 1$$

Подставим конкретные численные значения вместо переменных $x$ и $y$.

In [19]:
a=a.subs({x:1,y:2})
a
Out[19]:
$$- \frac{5}{3}$$

А сколько это будет численно?

In [20]:
a.n()
Out[20]:
$$-1.66666666666667$$

Элементарные функции

SymPy автоматически применяет упрощения элементарных функция (которые справедливы во всех случаях).

In [21]:
sin(-x)
Out[21]:
$$- \sin{\left (x \right )}$$
In [22]:
cos(pi/4),tan(5*pi/6)
Out[22]:
$$\left ( \frac{\sqrt{2}}{2}, \quad - \frac{\sqrt{3}}{3}\right )$$

SymPy может работать с числами с плавающей точкой, имеющими сколь угодно большую точность. Вот $\pi$ с 100 значащими цифрами.

In [23]:
pi.n(100)
Out[23]:
$$3.141592653589793238462643383279502884197169399375105820974944592307816406286208998628034825342117068$$

E - это основание натуральных логарифмов.

In [24]:
log(1),log(E)
Out[24]:
$$\left ( 0, \quad 1\right )$$
In [25]:
exp(log(x)),log(exp(x))
Out[25]:
$$\left ( x, \quad \log{\left (e^{x} \right )}\right )$$

А почему не $x$? Попробуйте подставить $x=2\pi i$.

In [26]:
sqrt(0)
Out[26]:
$$0$$
In [27]:
sqrt(x)**4,sqrt(x**4)
Out[27]:
$$\left ( x^{2}, \quad \sqrt{x^{4}}\right )$$

А почему не $x^2$? Попробуйте подставить $x=i$.

Символы могут иметь некоторые свойства. Например, они могут быть положительными. Тогда SymPy может сильнее упростить квадратные корни.

In [28]:
p,q=symbols('p q',positive=True)
sqrt(p**2)
Out[28]:
$$p$$
In [29]:
sqrt(12*x**2*y),sqrt(12*p**2*y)
Out[29]:
$$\left ( 2 \sqrt{3} \sqrt{x^{2} y}, \quad 2 \sqrt{3} p \sqrt{y}\right )$$

Пусть символ $n$ будет целым (I - это мнимая единица).

In [30]:
n=Symbol('n',integer=True)
simplify(exp(2*pi*I*n))
Out[30]:
$$1$$
In [31]:
sin(pi*n)
Out[31]:
$$0$$

Метод rewrite пытается переписать выражение в терминах заданной функции.

In [32]:
cos(x).rewrite(exp),exp(I*x).rewrite(cos)
Out[32]:
$$\left ( \frac{e^{i x}}{2} + \frac{1}{2} e^{- i x}, \quad i \sin{\left (x \right )} + \cos{\left (x \right )}\right )$$
In [33]:
asin(x).rewrite(log)
Out[33]:
$$- i \log{\left (i x + \sqrt{- x^{2} + 1} \right )}$$

Функция trigsimp пытается переписать тригонометрическое выражение в наиболее простом виде. В программах лучше использовать более специализированные функции.

In [34]:
trigsimp(2*sin(x)**2+3*cos(x)**2)
Out[34]:
$$\cos^{2}{\left (x \right )} + 2$$

Функция expand_trig разлагает синусы и косинусы сумм и кратных углов.

In [35]:
expand_trig(sin(x-y)),expand_trig(sin(2*x))
Out[35]:
$$\left ( \sin{\left (x \right )} \cos{\left (y \right )} - \sin{\left (y \right )} \cos{\left (x \right )}, \quad 2 \sin{\left (x \right )} \cos{\left (x \right )}\right )$$

Чаще нужно обратное преобразование - произведений и степеней синусов и косинусов в выражения, линейные по этим функциям. Например, пусть мы работаем с отрезком ряда Фурье.

In [36]:
a1,a2,b1,b2=symbols('a1 a2 b1 b2')
a=a1*cos(x)+a2*cos(2*x)+b1*sin(x)+b2*sin(2*x)
a
Out[36]:
$$a_{1} \cos{\left (x \right )} + a_{2} \cos{\left (2 x \right )} + b_{1} \sin{\left (x \right )} + b_{2} \sin{\left (2 x \right )}$$

Мы хотим возвести его в квадрат и опять получить отрезок ряда Фурье.

In [37]:
a=(a**2).rewrite(exp).expand().rewrite(cos).expand()
a
Out[37]:
$$\frac{a_{1}^{2}}{2} \cos{\left (2 x \right )} + \frac{a_{1}^{2}}{2} + a_{1} a_{2} \cos{\left (x \right )} + a_{1} a_{2} \cos{\left (3 x \right )} + a_{1} b_{1} \sin{\left (2 x \right )} + a_{1} b_{2} \sin{\left (x \right )} + a_{1} b_{2} \sin{\left (3 x \right )} + \frac{a_{2}^{2}}{2} \cos{\left (4 x \right )} + \frac{a_{2}^{2}}{2} - a_{2} b_{1} \sin{\left (x \right )} + a_{2} b_{1} \sin{\left (3 x \right )} + a_{2} b_{2} \sin{\left (4 x \right )} - \frac{b_{1}^{2}}{2} \cos{\left (2 x \right )} + \frac{b_{1}^{2}}{2} + b_{1} b_{2} \cos{\left (x \right )} - b_{1} b_{2} \cos{\left (3 x \right )} - \frac{b_{2}^{2}}{2} \cos{\left (4 x \right )} + \frac{b_{2}^{2}}{2}$$
In [38]:
a.collect([cos(x),cos(2*x),cos(3*x),sin(x),sin(2*x),sin(3*x)])
Out[38]:
$$\frac{a_{1}^{2}}{2} + a_{1} b_{1} \sin{\left (2 x \right )} + \frac{a_{2}^{2}}{2} \cos{\left (4 x \right )} + \frac{a_{2}^{2}}{2} + a_{2} b_{2} \sin{\left (4 x \right )} + \frac{b_{1}^{2}}{2} - \frac{b_{2}^{2}}{2} \cos{\left (4 x \right )} + \frac{b_{2}^{2}}{2} + \left(\frac{a_{1}^{2}}{2} - \frac{b_{1}^{2}}{2}\right) \cos{\left (2 x \right )} + \left(a_{1} a_{2} - b_{1} b_{2}\right) \cos{\left (3 x \right )} + \left(a_{1} a_{2} + b_{1} b_{2}\right) \cos{\left (x \right )} + \left(a_{1} b_{2} - a_{2} b_{1}\right) \sin{\left (x \right )} + \left(a_{1} b_{2} + a_{2} b_{1}\right) \sin{\left (3 x \right )}$$

Функция expand_log преобразует логарифмы произведений и степеней в суммы логарифмов (только для положительных величин); logcombine производит обратное преобразование.

In [39]:
a=expand_log(log(p*q**2))
a
Out[39]:
$$\log{\left (p \right )} + 2 \log{\left (q \right )}$$
In [40]:
logcombine(a)
Out[40]:
$$\log{\left (p q^{2} \right )}$$

Функция expand_power_exp переписывает степени, показатели которых - суммы, через произведения степеней.

In [41]:
expand_power_exp(x**(p+q))
Out[41]:
$$x^{p} x^{q}$$

Функция expand_power_base переписывает степени, основания которых - произведения, через произведения степеней.

In [42]:
expand_power_base((x*y)**n)
Out[42]:
$$x^{n} y^{n}$$

Функция powsimp выполняет обратные преобразования.

In [43]:
powsimp(exp(x)*exp(2*y)),powsimp(x**n*y**n)
Out[43]:
$$\left ( e^{x + 2 y}, \quad \left(x y\right)^{n}\right )$$

Можно вводить функции пользователя. Они могут иметь произвольное число аргументов.

In [44]:
f=Function('f')
f(x)+f(x,y)
Out[44]:
$$f{\left (x \right )} + f{\left (x,y \right )}$$

Структура выражений

Внутреннее представление выражения - это дерево. Функция srepr возвращает строку, представляющую его.

In [45]:
srepr(x+1)
Out[45]:
"Add(Symbol('x'), Integer(1))"
In [46]:
srepr(x-1)
Out[46]:
"Add(Symbol('x'), Integer(-1))"
In [47]:
srepr(x-y)
Out[47]:
"Add(Symbol('x'), Mul(Integer(-1), Symbol('y')))"
In [48]:
srepr(2*x*y/3)
Out[48]:
"Mul(Rational(2, 3), Symbol('x'), Symbol('y'))"
In [49]:
srepr(x/y)
Out[49]:
"Mul(Symbol('x'), Pow(Symbol('y'), Integer(-1)))"

Вместо бинарных операций +, *, ** и т.д. можно использовать функции Add, Mul, Pow и т.д.

In [50]:
Mul(x,Pow(y,-1))==x/y
Out[50]:
True
In [51]:
srepr(f(x,y))
Out[51]:
"Function('f')(Symbol('x'), Symbol('y'))"

Атрибут func - это функция верхнего уровня в выражении, а args - список её аргументов.

In [52]:
a=2*x*y**2
a.func
Out[52]:
sympy.core.mul.Mul
In [53]:
a.args
Out[53]:
$$\left ( 2, \quad x, \quad y^{2}\right )$$
In [54]:
a.args[0]
Out[54]:
$$2$$
In [55]:
for i in a.args:
    print(i)
2
x
y**2

Функция subs заменяет переменную на выражение.

In [56]:
a.subs(y,2)
Out[56]:
$$8 x$$

Она можнет заменить несколько переменных. Для этого ей передаётся список кортежей или словарь.

In [57]:
a.subs([(x,pi),(y,2)])
Out[57]:
$$8 \pi$$
In [58]:
a.subs({x:pi,y:2})
Out[58]:
$$8 \pi$$

Она может заменить не переменную, а подвыражение - функцию с аргументами.

In [59]:
a=f(x)+f(y)
a.subs(f(y),1)
Out[59]:
$$f{\left (x \right )} + 1$$
In [60]:
(2*x*y*z).subs(x*y,z)
Out[60]:
$$2 z^{2}$$
In [61]:
(x+x**2+x**3+x**4).subs(x**2,y)
Out[61]:
$$x^{3} + x + y^{2} + y$$

Подстановки производятся последовательно. В данном случае сначала $x$ заменился на $y$, получилось $y^3+y^2$; потом в этом результате $y$ заменилось на $x$.

In [62]:
a=x**2+y**3
a.subs([(x,y),(y,x)])
Out[62]:
$$x^{3} + x^{2}$$

Если написать эти подстановки в другом порядке, результат будет другим.

In [63]:
a.subs([(y,x),(x,y)])
Out[63]:
$$y^{3} + y^{2}$$

Но можно передать функции subs ключевой параметр simultaneous=True, тогда подстановки будут производиться одновременно. Таким образом можно, например, поменять местами $x$ и $y$.

In [64]:
a.subs([(x,y),(y,x)],simultaneous=True)
Out[64]:
$$x^{3} + y^{2}$$

Можно заменить функцию на другую функцию.

In [65]:
g=Function('g')
a=f(x)+f(y)
a.subs(f,g)
Out[65]:
$$g{\left (x \right )} + g{\left (y \right )}$$

Метод replace ищет подвыражения, соответствующие образцу (содержащему произвольные переменные), и заменяет их на заданное выражение (оно может содержать те же произвольные переменные).

In [66]:
a=Wild('a')
(f(x)+f(x+y)).replace(f(a),a**2)
Out[66]:
$$x^{2} + \left(x + y\right)^{2}$$
In [67]:
(f(x,x)+f(x,y)).replace(f(a,a),a**2)
Out[67]:
$$x^{2} + f{\left (x,y \right )}$$
In [68]:
a=x**2+y**2
a.replace(x,x+1)
Out[68]:
$$y^{2} + \left(x + 1\right)^{2}$$

Соответствовать образцу должно целое подвыражение, это не может быть часть сомножителей в произведении или меньшая степеть в большей.

In [69]:
a=2*x*y*z
a.replace(x*y,z)
Out[69]:
$$2 x y z$$
In [70]:
(x+x**2+x**3+x**4).replace(x**2,y)
Out[70]:
$$x^{4} + x^{3} + x + y$$

Решение уравнений

In [71]:
a,b,c,d,e,f=symbols('a b c d e f')

Уравнение записывается как функция Eq с двумя параметрами. Функция solve возврящает список решений.

In [72]:
solve(Eq(a*x,b),x)
Out[72]:
$$\left [ \frac{b}{a}\right ]$$

Впрочем, можно передать функции solve просто выражение. Подразумевается уравнение, что это выражение равно 0.

In [73]:
solve(a*x+b,x)
Out[73]:
$$\left [ - \frac{b}{a}\right ]$$

Квадратное уравнение имеет 2 решения.

In [74]:
solve(a*x**2+b*x+c,x)
Out[74]:
$$\left [ \frac{1}{2 a} \left(- b + \sqrt{- 4 a c + b^{2}}\right), \quad - \frac{1}{2 a} \left(b + \sqrt{- 4 a c + b^{2}}\right)\right ]$$

Система линейных уравнений.

In [75]:
solve([a*x+b*y-e,c*x+d*y-f],[x,y])
Out[75]:
$$\left \{ x : \frac{- b f + d e}{a d - b c}, \quad y : \frac{a f - c e}{a d - b c}\right \}$$

Функция roots возвращает корни многочлена с их множественностями.

In [76]:
roots(x**3-3*x+2,x)
Out[76]:
$$\left \{ -2 : 1, \quad 1 : 2\right \}$$

Функция solve_poly_system решает систему полиномиальных уравнений, строя их базис Грёбнера.

In [77]:
p1=x**2+y**2-1
p2=4*x*y-1
solve_poly_system([p1,p2],x,y)
Out[77]:
$$\left [ \left ( 4 \left(-1 - \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}}\right) \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}} \left(- \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}} + 1\right), \quad - \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}}\right ), \quad \left ( - 4 \left(-1 + \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}}\right) \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}} \left(\sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}} + 1\right), \quad \sqrt{- \frac{\sqrt{3}}{4} + \frac{1}{2}}\right ), \quad \left ( 4 \left(-1 - \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}}\right) \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}} \left(- \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}} + 1\right), \quad - \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}}\right ), \quad \left ( - 4 \left(-1 + \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}}\right) \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}} \left(\sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}} + 1\right), \quad \sqrt{\frac{\sqrt{3}}{4} + \frac{1}{2}}\right )\right ]$$

Ряды

In [78]:
exp(x).series(x,0,5)
Out[78]:
$$1 + x + \frac{x^{2}}{2} + \frac{x^{3}}{6} + \frac{x^{4}}{24} + \mathcal{O}\left(x^{5}\right)$$

Ряд может начинаться с отрицательной степени.

In [79]:
cot(x).series(x,n=5)
Out[79]:
$$\frac{1}{x} - \frac{x}{3} - \frac{x^{3}}{45} + \mathcal{O}\left(x^{5}\right)$$

И даже идти по полуцелым степеням.

In [80]:
sqrt(x*(1-x)).series(x,n=5)
Out[80]:
$$\sqrt{x} - \frac{x^{\frac{3}{2}}}{2} - \frac{x^{\frac{5}{2}}}{8} - \frac{x^{\frac{7}{2}}}{16} - \frac{5 x^{\frac{9}{2}}}{128} + \mathcal{O}\left(x^{5}\right)$$
In [81]:
log(gamma(1+x)).series(x,n=6).rewrite(zeta)
Out[81]:
$$- \gamma x + \frac{\pi^{2} x^{2}}{12} - \frac{x^{3} \zeta\left(3\right)}{3} + \frac{\pi^{4} x^{4}}{360} - \frac{x^{5} \zeta\left(5\right)}{5} + \mathcal{O}\left(x^{6}\right)$$

Подготовим 3 ряда.

In [82]:
sinx=series(sin(x),x,0,8)
sinx
Out[82]:
$$x - \frac{x^{3}}{6} + \frac{x^{5}}{120} - \frac{x^{7}}{5040} + \mathcal{O}\left(x^{8}\right)$$
In [83]:
cosx=series(cos(x),x,n=8)
cosx
Out[83]:
$$1 - \frac{x^{2}}{2} + \frac{x^{4}}{24} - \frac{x^{6}}{720} + \mathcal{O}\left(x^{8}\right)$$
In [84]:
tanx=series(tan(x),x,n=8)
tanx
Out[84]:
$$x + \frac{x^{3}}{3} + \frac{2 x^{5}}{15} + \frac{17 x^{7}}{315} + \mathcal{O}\left(x^{8}\right)$$

Произведения и частные рядов не вычисляются автоматически, к ним надо применить функцию series.

In [85]:
series(tanx*cosx,n=8)
Out[85]:
$$x - \frac{x^{3}}{6} + \frac{x^{5}}{120} - \frac{x^{7}}{5040} + \mathcal{O}\left(x^{8}\right)$$
In [86]:
series(sinx/cosx,n=8)
Out[86]:
$$x + \frac{x^{3}}{3} + \frac{2 x^{5}}{15} + \frac{17 x^{7}}{315} + \mathcal{O}\left(x^{8}\right)$$

А этот ряд должен быть равен 1. Но поскольку sinx и cosx известны лишь с ограниченной точностью, мы получаем 1 с той же точностью.

In [87]:
series(sinx**2+cosx**2,n=8)
Out[87]:
$$1 + \mathcal{O}\left(x^{8}\right)$$

Здесь первые члены сократились, и ответ можно получить лишь с меньшей точностью.

In [88]:
series((1-cosx)/x**2,n=6)
Out[88]:
$$\frac{1}{2} - \frac{x^{2}}{24} + \frac{x^{4}}{720} + \mathcal{O}\left(x^{6}\right)$$

Ряды можно дифференцировать и интегрировать.

In [89]:
diff(sinx,x)
Out[89]:
$$1 - \frac{x^{2}}{2} + \frac{x^{4}}{24} - \frac{x^{6}}{720} + \mathcal{O}\left(x^{7}\right)$$
In [90]:
integrate(cosx,x)
Out[90]:
$$x - \frac{x^{3}}{6} + \frac{x^{5}}{120} - \frac{x^{7}}{5040} + \mathcal{O}\left(x^{9}\right)$$

Можно подставить ряд (если он начинается с малого члена) вместо переменной разложения в другой ряд. Вот ряды для $\sin(\tan(x))$ и $\tan(\sin(x))$.

In [91]:
st=series(sinx.subs(x,tanx),n=8)
st
Out[91]:
$$x + \frac{x^{3}}{6} - \frac{x^{5}}{40} - \frac{55 x^{7}}{1008} + \mathcal{O}\left(x^{8}\right)$$
In [92]:
ts=series(tanx.subs(x,sinx),n=8)
ts
Out[92]:
$$x + \frac{x^{3}}{6} - \frac{x^{5}}{40} - \frac{107 x^{7}}{5040} + \mathcal{O}\left(x^{8}\right)$$
In [93]:
series(ts-st,n=8)
Out[93]:
$$\frac{x^{7}}{30} + \mathcal{O}\left(x^{8}\right)$$

В ряд нельзя подставлять численное значение переменной разложения (а значит, нельзя и строить график). Для этого нужно сначала убрать $\mathcal{O}$ член, превратив отрезок ряда в многочлен.

In [94]:
a=sinx.removeO()
In [95]:
a.subs(x,0.1)
Out[95]:
$$0.0998334166468254$$

Производные

In [96]:
a=x*sin(x+y)
diff(a,x)
Out[96]:
$$x \cos{\left (x + y \right )} + \sin{\left (x + y \right )}$$
In [97]:
diff(a,y)
Out[97]:
$$x \cos{\left (x + y \right )}$$

Вторая производная по $x$ и первая по $y$.

In [98]:
diff(a,x,2,y)
Out[98]:
$$- x \cos{\left (x + y \right )} + 2 \sin{\left (x + y \right )}$$

Можно дифференцировать выражения, содержащие неопределённые функции.

In [99]:
a=x*f(x**2)
b=diff(a,x)
b
Out[99]:
$$2 x^{2} \left. \frac{d}{d \xi_{1}} f{\left (\xi_{1} \right )} \right|_{\substack{ \xi_{1}=x^{2} }} + f{\left (x^{2} \right )}$$

Что это за зверь такой получился?

In [100]:
print(b)
2*x**2*Subs(Derivative(f(_xi_1), _xi_1), (_xi_1,), (x**2,)) + f(x**2)

Функция Derivative представляет невычисленную производную. Её можно вычислить методом doit.

In [101]:
a=Derivative(sin(x),x)
Eq(a,a.doit())
Out[101]:
$$\frac{d}{d x} \sin{\left (x \right )} = \cos{\left (x \right )}$$

Интегралы

In [102]:
integrate(1/(x*(x**2-2)**2),x)
Out[102]:
$$\frac{1}{4} \log{\left (x \right )} - \frac{1}{8} \log{\left (x^{2} - 2 \right )} - \frac{1}{4 x^{2} - 8}$$
In [103]:
integrate(1/(exp(x)+1),x)
Out[103]:
$$x - \log{\left (e^{x} + 1 \right )}$$
In [104]:
integrate(log(x),x)
Out[104]:
$$x \log{\left (x \right )} - x$$
In [105]:
integrate(x*sin(x),x)
Out[105]:
$$- x \cos{\left (x \right )} + \sin{\left (x \right )}$$
In [106]:
integrate(x*exp(-x**2),x)
Out[106]:
$$- \frac{e^{- x^{2}}}{2}$$
In [107]:
a=integrate(x**x,x)
a
Out[107]:
$$\int x^{x}\, dx$$

Получился невычисленный интеграл.

In [108]:
print(a)
Integral(x**x, x)
In [109]:
a=Integral(sin(x),x)
Eq(a,a.doit())
Out[109]:
$$\int \sin{\left (x \right )}\, dx = - \cos{\left (x \right )}$$

Определённые интегралы.

In [110]:
integrate(sin(x),(x,0,pi))
Out[110]:
$$2$$

oo - это $\infty$.

In [111]:
integrate(exp(-x**2),(x,0,oo))
Out[111]:
$$\frac{\sqrt{\pi}}{2}$$
In [112]:
integrate(log(x)/(1-x),(x,0,1))
Out[112]:
$$- \frac{\pi^{2}}{6}$$

Суммирование рядов

In [113]:
summation(1/n**2,(n,1,oo))
Out[113]:
$$\frac{\pi^{2}}{6}$$
In [114]:
summation((-1)**n/n**2,(n,1,oo))
Out[114]:
$$- \frac{\pi^{2}}{12}$$
In [115]:
summation(1/n**4,(n,1,oo))
Out[115]:
$$\frac{\pi^{4}}{90}$$

Невычисленная сумма обозначается Sum.

In [116]:
a=Sum(x**n/factorial(n),(n,0,oo))
Eq(a,a.doit())
Out[116]:
$$\sum_{n=0}^{\infty} \frac{x^{n}}{n!} = e^{x}$$

Пределы

In [117]:
limit((tan(sin(x))-sin(tan(x)))/x**7,x,0)
Out[117]:
$$\frac{1}{30}$$

Ну это простой предел, считается разложением числителя и знаменателя в ряды. А вот если в $x=0$ существенно особая точка, дело сложнее. Посчитаем односторонние пределы.

In [118]:
limit((tan(sin(x))-sin(tan(x)))/(x**7+exp(-1/x)),x,0,'+')
Out[118]:
$$\frac{1}{30}$$
In [119]:
limit((tan(sin(x))-sin(tan(x)))/(x**7+exp(-1/x)),x,0,'-')
Out[119]:
$$0$$

Дифференциальные уравнения

In [120]:
t=Symbol('t')
x=Function('x')
p=Function('p')

Первого порядка.

In [121]:
dsolve(diff(x(t),t)+x(t),x(t))
Out[121]:
$$x{\left (t \right )} = C_{1} e^{- t}$$

Второго порядка.

In [122]:
dsolve(diff(x(t),t,2)+x(t),x(t))
Out[122]:
$$x{\left (t \right )} = C_{1} \sin{\left (t \right )} + C_{2} \cos{\left (t \right )}$$

Система уравнений первого порядка.

In [123]:
dsolve((diff(x(t),t)-p(t),diff(p(t),t)+x(t)))
Out[123]:
$$\left [ x{\left (t \right )} = C_{1} \sin{\left (t \right )} + C_{2} \cos{\left (t \right )}, \quad p{\left (t \right )} = C_{1} \cos{\left (t \right )} - C_{2} \sin{\left (t \right )}\right ]$$

Линейная алгебра

In [124]:
a,b,c,d,e,f=symbols('a b c d e f')

Матрицу можно построить из списка списков.

In [125]:
M=Matrix([[a,b,c],[d,e,f]])
M
Out[125]:
$$\left[\begin{matrix}a & b & c\\d & e & f\end{matrix}\right]$$
In [126]:
M.shape
Out[126]:
$$\left ( 2, \quad 3\right )$$

Матрица-строка.

In [127]:
Matrix([[1,2,3]])
Out[127]:
$$\left[\begin{matrix}1 & 2 & 3\end{matrix}\right]$$

Матрица-столбец.

In [128]:
Matrix([1,2,3])
Out[128]:
$$\left[\begin{matrix}1\\2\\3\end{matrix}\right]$$

Можно построить матрицу из функции.

In [129]:
def g(i,j):
    return Rational(1,i+j+1)
Matrix(3,3,g)
Out[129]:
$$\left[\begin{matrix}1 & \frac{1}{2} & \frac{1}{3}\\\frac{1}{2} & \frac{1}{3} & \frac{1}{4}\\\frac{1}{3} & \frac{1}{4} & \frac{1}{5}\end{matrix}\right]$$

Или из неопределённой функции.

In [130]:
g=Function('g')
M=Matrix(3,3,g)
M
Out[130]:
$$\left[\begin{matrix}g{\left (0,0 \right )} & g{\left (0,1 \right )} & g{\left (0,2 \right )}\\g{\left (1,0 \right )} & g{\left (1,1 \right )} & g{\left (1,2 \right )}\\g{\left (2,0 \right )} & g{\left (2,1 \right )} & g{\left (2,2 \right )}\end{matrix}\right]$$
In [131]:
M[1,2]
Out[131]:
$$g{\left (1,2 \right )}$$
In [132]:
M[1,2]=0
M
Out[132]:
$$\left[\begin{matrix}g{\left (0,0 \right )} & g{\left (0,1 \right )} & g{\left (0,2 \right )}\\g{\left (1,0 \right )} & g{\left (1,1 \right )} & 0\\g{\left (2,0 \right )} & g{\left (2,1 \right )} & g{\left (2,2 \right )}\end{matrix}\right]$$
In [133]:
M[2,:]
Out[133]:
$$\left[\begin{matrix}g{\left (2,0 \right )} & g{\left (2,1 \right )} & g{\left (2,2 \right )}\end{matrix}\right]$$
In [134]:
M[:,1]
Out[134]:
$$\left[\begin{matrix}g{\left (0,1 \right )}\\g{\left (1,1 \right )}\\g{\left (2,1 \right )}\end{matrix}\right]$$
In [135]:
M[0:2,1:3]
Out[135]:
$$\left[\begin{matrix}g{\left (0,1 \right )} & g{\left (0,2 \right )}\\g{\left (1,1 \right )} & 0\end{matrix}\right]$$

Единичная матрица.

In [136]:
eye(3)
Out[136]:
$$\left[\begin{matrix}1 & 0 & 0\\0 & 1 & 0\\0 & 0 & 1\end{matrix}\right]$$

Матрица из нулей.

In [137]:
zeros(3)
Out[137]:
$$\left[\begin{matrix}0 & 0 & 0\\0 & 0 & 0\\0 & 0 & 0\end{matrix}\right]$$
In [138]:
zeros(2,3)
Out[138]:
$$\left[\begin{matrix}0 & 0 & 0\\0 & 0 & 0\end{matrix}\right]$$

Диагональная матрица.

In [139]:
diag(1,2,3)
Out[139]:
$$\left[\begin{matrix}1 & 0 & 0\\0 & 2 & 0\\0 & 0 & 3\end{matrix}\right]$$
In [140]:
M=Matrix([[a,1],[0,a]])
diag(1,M,2)
Out[140]:
$$\left[\begin{matrix}1 & 0 & 0 & 0\\0 & a & 1 & 0\\0 & 0 & a & 0\\0 & 0 & 0 & 2\end{matrix}\right]$$

Операции с матрицами.

In [141]:
A=Matrix([[a,b],[c,d]])
B=Matrix([[1,2],[3,4]])
A+B
Out[141]:
$$\left[\begin{matrix}a + 1 & b + 2\\c + 3 & d + 4\end{matrix}\right]$$
In [142]:
A*B,B*A
Out[142]:
$$\left ( \left[\begin{matrix}a + 3 b & 2 a + 4 b\\c + 3 d & 2 c + 4 d\end{matrix}\right], \quad \left[\begin{matrix}a + 2 c & b + 2 d\\3 a + 4 c & 3 b + 4 d\end{matrix}\right]\right )$$
In [143]:
A*B-B*A
Out[143]:
$$\left[\begin{matrix}3 b - 2 c & 2 a + 3 b - 2 d\\- 3 a - 3 c + 3 d & - 3 b + 2 c\end{matrix}\right]$$
In [144]:
simplify(A**(-1))
Out[144]:
$$\left[\begin{matrix}\frac{d}{a d - b c} & - \frac{b}{a d - b c}\\- \frac{c}{a d - b c} & \frac{a}{a d - b c}\end{matrix}\right]$$
In [145]:
det(A)
Out[145]:
$$a d - b c$$

Собственные значения и векторы

In [146]:
x=Symbol('x',real=True)
In [147]:
M=Matrix([[(1-x)**3*(3+x),4*x*(1-x**2),-2*(1-x**2)*(3-x)],
          [4*x*(1-x**2),-(1+x)**3*(3-x),2*(1-x**2)*(3+x)],
          [-2*(1-x**2)*(3-x),2*(1-x**2)*(3+x),16*x]])
M
Out[147]:
$$\left[\begin{matrix}\left(- x + 1\right)^{3} \left(x + 3\right) & 4 x \left(- x^{2} + 1\right) & \left(- x + 3\right) \left(2 x^{2} - 2\right)\\4 x \left(- x^{2} + 1\right) & - \left(- x + 3\right) \left(x + 1\right)^{3} & \left(x + 3\right) \left(- 2 x^{2} + 2\right)\\\left(- x + 3\right) \left(2 x^{2} - 2\right) & \left(x + 3\right) \left(- 2 x^{2} + 2\right) & 16 x\end{matrix}\right]$$
In [148]:
det(M)
Out[148]:
$$0$$

Значит, у этой матрицы есть нулевое подпространство (она обращает векторы из этого подпространства в 0). Базис этого подпространства.

In [149]:
v=M.nullspace()
len(v)
Out[149]:
$$1$$

Оно одномерно.

In [150]:
v=simplify(v[0])
v
Out[150]:
$$\left[\begin{matrix}- \frac{2}{x - 1}\\\frac{2}{x + 1}\\1\end{matrix}\right]$$

Проверим.

In [151]:
simplify(M*v)
Out[151]:
$$\left[\begin{matrix}0\\0\\0\end{matrix}\right]$$

Собственные значения и их кратности.

In [152]:
M.eigenvals()
Out[152]:
$$\left \{ 0 : 1, \quad - \left(x^{2} + 3\right)^{2} : 1, \quad \left(x^{2} + 3\right)^{2} : 1\right \}$$

Если нужны не только собственные значения, но и собственные векторы, то нужно использовать метод eigenvects. Он возвращает список кортежей. В каждом из них нулевой элемент - собственное значение, первый - его кратность, и последний - список собственных векторов, образующих базис (их столько, какова кратность).

In [153]:
v=M.eigenvects()
len(v)
Out[153]:
$$3$$
In [154]:
for i in range(len(v)):
    v[i][2][0]=simplify(v[i][2][0])
v
Out[154]:
$$\left [ \left ( 0, \quad 1, \quad \left [ \left[\begin{matrix}- \frac{2}{x - 1}\\\frac{2}{x + 1}\\1\end{matrix}\right]\right ]\right ), \quad \left ( - \left(x^{2} + 3\right)^{2}, \quad 1, \quad \left [ \left[\begin{matrix}\frac{x}{2} + \frac{1}{2}\\\frac{x + 1}{x - 1}\\1\end{matrix}\right]\right ]\right ), \quad \left ( \left(x^{2} + 3\right)^{2}, \quad 1, \quad \left [ \left[\begin{matrix}\frac{x - 1}{x + 1}\\- \frac{x}{2} + \frac{1}{2}\\1\end{matrix}\right]\right ]\right )\right ]$$

Проверим.

In [155]:
for i in range(len(v)):
    z=M*v[i][2][0]-v[i][0]*v[i][2][0]
    pprint(simplify(z))
⎡0⎤
⎢ ⎥
⎢0⎥
⎢ ⎥
⎣0⎦
⎡0⎤
⎢ ⎥
⎢0⎥
⎢ ⎥
⎣0⎦
⎡0⎤
⎢ ⎥
⎢0⎥
⎢ ⎥
⎣0⎦

Жорданова нормальная форма

In [156]:
M=Matrix([[Rational(13,9),-Rational(2,9),Rational(1,3),Rational(4,9),Rational(2,3)],
          [-Rational(2,9),Rational(10,9),Rational(2,15),-Rational(2,9),-Rational(11,15)],
          [Rational(1,5),-Rational(2,5),Rational(41,25),-Rational(2,5),Rational(12,25)],
          [Rational(4,9),-Rational(2,9),Rational(14,15),Rational(13,9),-Rational(2,15)],
          [-Rational(4,15),Rational(8,15),Rational(12,25),Rational(8,15),Rational(34,25)]])
M
Out[156]:
$$\left[\begin{matrix}\frac{13}{9} & - \frac{2}{9} & \frac{1}{3} & \frac{4}{9} & \frac{2}{3}\\- \frac{2}{9} & \frac{10}{9} & \frac{2}{15} & - \frac{2}{9} & - \frac{11}{15}\\\frac{1}{5} & - \frac{2}{5} & \frac{41}{25} & - \frac{2}{5} & \frac{12}{25}\\\frac{4}{9} & - \frac{2}{9} & \frac{14}{15} & \frac{13}{9} & - \frac{2}{15}\\- \frac{4}{15} & \frac{8}{15} & \frac{12}{25} & \frac{8}{15} & \frac{34}{25}\end{matrix}\right]$$

Метод M.jordan_form() возвращает пару матриц, матрицу преобразования $P$ и собственно жорданову форму $J$: $M = P J P^{-1}$.

In [157]:
P,J=M.jordan_form()
J
Out[157]:
$$\left[\begin{matrix}1 & 0 & 0 & 0 & 0\\0 & 2 & 1 & 0 & 0\\0 & 0 & 2 & 0 & 0\\0 & 0 & 0 & 1 - i & 0\\0 & 0 & 0 & 0 & 1 + i\end{matrix}\right]$$
In [158]:
P=simplify(P)
P
Out[158]:
$$\left[\begin{matrix}-2 & \frac{10}{9} & 0 & \frac{5 i}{12} & - \frac{5 i}{12}\\-2 & - \frac{5}{9} & 0 & - \frac{5 i}{6} & \frac{5 i}{6}\\0 & 0 & \frac{4}{3} & - \frac{3}{4} & - \frac{3}{4}\\1 & \frac{10}{9} & 0 & - \frac{5 i}{6} & \frac{5 i}{6}\\0 & 0 & 1 & 1 & 1\end{matrix}\right]$$

Проверим.

In [159]:
Z=P*J*P**(-1)-M
simplify(Z)
Out[159]:
$$\left[\begin{matrix}0 & 0 & 0 & 0 & 0\\0 & 0 & 0 & 0 & 0\\0 & 0 & 0 & 0 & 0\\0 & 0 & 0 & 0 & 0\\0 & 0 & 0 & 0 & 0\end{matrix}\right]$$

Графики

SymPy использует matplotlib. Однако он распределяет точки по $x$ адаптивно, а не равномерно.

In [160]:
%matplotlib inline

Одна функция.

In [161]:
plot(sin(x)/x,(x,-10,10))
Out[161]:
<sympy.plotting.plot.Plot at 0x7fac5f8ff6d8>

Несколько функций.

In [162]:
plot(x,x**2,x**3,(x,0,2))
Out[162]:
<sympy.plotting.plot.Plot at 0x7fac5f567390>

Другие функции надо импортировать из пакета sympy.plotting.

In [163]:
from sympy.plotting import (plot_parametric,plot_implicit,
                            plot3d,plot3d_parametric_line,
                            plot3d_parametric_surface)

Параметрический график - фигура Лиссажу.

In [164]:
t=Symbol('t')
plot_parametric(sin(2*t),cos(3*t),(t,0,2*pi),
                title='Lissajous',xlabel='x',ylabel='y')
Out[164]:
<sympy.plotting.plot.Plot at 0x7fac5f514978>

Неявный график - окружность.

In [165]:
plot_implicit(x**2+y**2-1,(x,-1,1),(y,-1,1))
Out[165]:
<sympy.plotting.plot.Plot at 0x7fac5f625940>

Поверхность. Если она строится не inline, а в отдельном окне, то её можно вертеть мышкой.

In [166]:
plot3d(x*y,(x,-2,2),(y,-2,2))
Out[166]:
<sympy.plotting.plot.Plot at 0x7fac71242ac8>

Несколько поверхностей.

In [167]:
plot3d(x**2+y**2,x*y,(x,-2,2),(y,-2,2))
Out[167]:
<sympy.plotting.plot.Plot at 0x7fac5f3ac9b0>

Параметрическая пространственная линия - спираль.

In [168]:
a=0.1
plot3d_parametric_line(cos(t),sin(t),a*t,(t,0,4*pi))
Out[168]:
<sympy.plotting.plot.Plot at 0x7fac5d25be10>

Параметрическая поверхность - тор.

In [169]:
u,v=symbols('u v')
a=0.4
plot3d_parametric_surface((1+a*cos(u))*cos(v),
                          (1+a*cos(u))*sin(v),a*sin(u),
                          (u,0,2*pi),(v,0,2*pi))
Out[169]:
<sympy.plotting.plot.Plot at 0x7fac5d151f60>