代数と微積分の基本¶
Sageでは,初等的な代数と微積分に関連した多様な演算を実行することができる. 例として,方程式の解を求める,微分や積分を計算する,ラプラス変換の実行などがあげられる. Sage Constructions には,さらに多様な具体例が盛られている.
方程式を解く¶
方程式を解析的に解く¶
solve
関数を使って方程式の解を求めることができる.
これを使うには、まず変数を定義し、ついで対象とする方程式(または方程式系)と解くべき変数を solve
の引数として指定する:
sage: x = var('x')
sage: solve(x^2 + 3*x + 2, x)
[x == -2, x == -1]
解くべき変数を変更して,解を他の変数で表わすこともできる:
sage: x, b, c = var('x b c')
sage: solve([x^2 + b*x + c == 0],x)
[x == -1/2*b - 1/2*sqrt(b^2 - 4*c), x == -1/2*b + 1/2*sqrt(b^2 - 4*c)]
多変数の方程式を解くことも可能だ:
sage: x, y = var('x, y')
sage: solve([x+y==6, x-y==4], x, y)
[[x == 5, y == 1]]
次のJason Groutによる例題では、Sageを使って連立非線形方程式を解く.まず,この連立方程式の解を記号的に求めてみよう:
sage: var('x y p q')
(x, y, p, q)
sage: eq1 = p+q==9
sage: eq2 = q*y+p*x==-6
sage: eq3 = q*y^2+p*x^2==24
sage: solve([eq1,eq2,eq3,p==1],p,q,x,y)
[[p == 1, q == 8, x == -4/3*sqrt(10) - 2/3, y == 1/6*sqrt(10) - 2/3], [p == 1, q == 8, x == 4/3*sqrt(10) - 2/3, y == -1/6*sqrt(10) - 2/3]]
解の数値近似を求めるには,やり方を変えて:
sage: solns = solve([eq1,eq2,eq3,p==1],p,q,x,y, solution_dict=True)
sage: [[s[p].n(30), s[q].n(30), s[x].n(30), s[y].n(30)] for s in solns]
[[1.0000000, 8.0000000, -4.8830369, -0.13962039],
[1.0000000, 8.0000000, 3.5497035, -1.1937129]]
n
関数は解の数値的近似値を表示する. n
の引数は数値精度を表わすビット数を指定している.
方程式を数値的に解く¶
目的の方程式(または方程式系)に対し solve
では厳密解を求めることができないというのは珍しいことではない.
そうした場合には find_root
を使って数値解を求めることができる.
例えば,以下に示す方程式については solve
は何も役に立つことを教えてくれない.
sage: theta = var('theta')
sage: solve(cos(theta)==sin(theta), theta)
[sin(theta) == cos(theta)]
しかし代りに find_root
を使えば,
sage: phi = var('phi')
sage: find_root(cos(phi)==sin(phi),0,pi/2)
0.785398163397448...
微分,積分,その他¶
Sageで多様な関数の微分と積分を計算することができる.
例えば
sage: u = var('u')
sage: diff(sin(u), u)
cos(u)
sage: diff(sin(x^2), x, 4)
16*x^4*sin(x^2) - 48*x^2*cos(x^2) - 12*sin(x^2)
sage: x, y = var('x,y')
sage: f = x^2 + 17*y^2
sage: f.diff(x)
2*x
sage: f.diff(y)
34*y
次は不定積分と定積分だ.
sage: integral(x*sin(x^2), x)
-1/2*cos(x^2)
sage: integral(x/(x^2+1), x, 0, 1)
1/2*log(2)
sage: f = 1/((1+x)*(x-1))
sage: f.partial_fraction(x)
-1/2/(x + 1) + 1/2/(x - 1)
微分方程式を解く¶
Sageを使って常微分方程式を研究することもできる.
sage: t = var('t') # 変数 t を定義
sage: x = function('x')(t) # x を t の関数とする
sage: DE = diff(x, t) + x - 1
sage: desolve(DE, [x,t])
(_C + e^t)*e^(-t)
ここでSageはMaxima [Max] とインターフェイスしているので,その出力もこれまで見てきたSageの出力とは若干違っている.
上の結果は,上の微分方程式の一般解が
ラプラス変換を実行することができる.
sage: s = var("s")
sage: t = var("t")
sage: f = t^2*exp(t) - sin(t)
sage: f.laplace(t,s)
-1/(s^2 + 1) + 2/(s - 1)^3
もう少し手間のかかる問題を考えてみよう. 左端が壁に固定された連成バネ各々の、平衡位置からの変位
|------\/\/\/\/\---|mass1|----\/\/\/\/\/----|mass2|
spring1 spring2
は、連立2階微分方程式
でモデル化される.
ここで
例題: 上の問題で各パラメータの値を
解法: まず1番目の方程式をラプラス変換する(記号は
sage: de1 = maxima("2*diff(x(t),t, 2) + 6*x(t) - 2*y(t)")
sage: lde1 = de1.laplace("t","s"); lde1.sage()
2*s^2*laplace(x(t), t, s) - 2*s*x(0) + 6*laplace(x(t), t, s) - 2*laplace(y(t), t, s) - 2*D[0](x)(0)
この出力は読みにくいけれども,意味しているのは
ということだ(ここでは小文字名の関数
sage: t,s = SR.var('t,s')
sage: x = function('x')
sage: y = function('y')
sage: f = 2*x(t).diff(t,2) + 6*x(t) - 2*y(t)
sage: f.laplace(t,s)
2*s^2*laplace(x(t), t, s) - 2*s*x(0) + 6*laplace(x(t), t, s) - 2*laplace(y(t), t, s) - 2*D[0](x)(0)
意味するところは
初期条件
sage: var('s X Y')
(s, X, Y)
sage: eqns = [(2*s^2+6)*X-2*Y == 6*s, -2*X +(s^2+2)*Y == 3*s]
sage: solve(eqns, X,Y)
[[X == 3*(s^3 + 3*s)/(s^4 + 5*s^2 + 4),
Y == 3*(s^3 + 5*s)/(s^4 + 5*s^2 + 4)]]
この解の逆ラプラス変換を行なうと:
sage: var('s t')
(s, t)
sage: inverse_laplace((3*s^3 + 9*s)/(s^4 + 5*s^2 + 4),s,t)
cos(2*t) + 2*cos(t)
sage: inverse_laplace((3*s^3 + 15*s)/(s^4 + 5*s^2 + 4),s,t)
-cos(2*t) + 4*cos(t)
というわけで,求めていた解は
これを媒介変数プロットするには
sage: t = var('t')
sage: P = parametric_plot((cos(2*t) + 2*cos(t), 4*cos(t) - cos(2*t) ),
....: (t, 0, 2*pi), rgbcolor=hue(0.9))
sage: show(P)
各成分ごとにプロットするには
sage: t = var('t')
sage: p1 = plot(cos(2*t) + 2*cos(t), (t,0, 2*pi), rgbcolor=hue(0.3))
sage: p2 = plot(4*cos(t) - cos(2*t), (t,0, 2*pi), rgbcolor=hue(0.6))
sage: show(p1 + p2)
プロットについては プロットする 節の,もう少し詳しい説明を見てほしい. 微分方程式については [NagleEtAl2004] の5.5節にもっと詳しい解説がある.
オイラーによる連立微分方程式の解法¶
次の例では,1階および2階微分方程式に対するオイラーの解法を具体的に解説する. 手始めに1階微分方程式に対する解法の基本的アイデアを復習しておこう.初期値問題が
のような形式で与えられており,
微分係数の定義から
ここで
(他にうまい呼び方も思いつかないので)
と表わすことができる.
ここで
... |
||
... |
||
... |
||
??? |
... |
我々の目標は,この表の空欄を上から一行づつ全て埋めていき,最終的に
連立微分方程式に対する解法もアイデアは似ている.
例題:
ここでは問題の2階常微分方程式を(
sage: t,x,y = PolynomialRing(RealField(10),3,"txy").gens()
sage: f = y; g = -x - y * t
sage: eulers_method_2x2(f,g, 0, 1, 0, 1/4, 1)
t x h*f(t,x,y) y h*g(t,x,y)
0 1 0.00 0 -0.25
1/4 1.0 -0.062 -0.25 -0.23
1/2 0.94 -0.12 -0.48 -0.17
3/4 0.82 -0.16 -0.66 -0.081
1 0.65 -0.18 -0.74 0.022
したがって,
点 eulers_method_2x2_plot
を使うが,その前に三つの成分(
sage: f = lambda z: z[2] # f(t,x,y) = y
sage: g = lambda z: -sin(z[1]) # g(t,x,y) = -sin(x)
sage: P = eulers_method_2x2_plot(f,g, 0.0, 0.75, 0.0, 0.1, 1.0)
この時点で, P
は2系列のプロットを保持していることになる.
P[0]
, および P[1]
である.
これら二つをプロットするには、次のようにする:
sage: show(P[0] + P[1])
(プロットの詳細については プロットする 節を参照.)
特殊関数¶
数種類の直交多項式と特殊関数が,PARI [GAP] およびMaxima [Max] を援用して実装されている. 詳細についてはSageレファレンスマニュアルの“Orthogonal polynomials"(直交多項式)と“Special functions"(特殊関数)を参照してほしい.
sage: x = polygen(QQ, 'x')
sage: chebyshev_U(2,x)
4*x^2 - 1
sage: bessel_I(1,1).n(250)
0.56515910399248502720769602760986330732889962162109200948029448947925564096
sage: bessel_I(1,1).n()
0.565159103992485
sage: bessel_I(2,1.1).n()
0.167089499251049
ここで注意したいのは,Sageではこれらの関数群が専ら数値計算に便利なようにラップ(wrap)されている点だ. 記号処理をする場合には,以下の例のようにMaximaインターフェイスをじかに呼び出してほしい.
sage: maxima.eval("f:bessel_y(v, w)")
'bessel_y(v,w)'
sage: maxima.eval("diff(f,w)")
'(bessel_y(v-1,w)-bessel_y(v+1,w))/2'