Simo K. Kivelä / 05.05.2003

Yhtälöiden ratkaiseminen

In[1]:=

Remove["Global`*"]

•Ensimmäisen ja toisen asteen yhtälöt; yleisemmin: polynomiyhtälöt

Luontevinta on aloittaa tallettamalla yhtälö jollekin (kuvaavalle) nimelle. Yhtälön yhtäläisyysmerkki on kaksi peräkkäistä yhtäläisyysmerkkiä; yksi yhtäläisyysmerkki on sijoitusoperaattori, jolla lauseke talletetaan jollekin  nimelle.

In[2]:=

yhtalo = 17 x - 2 == 9

Out[2]=

-2 + 17 x == 9

Ratkaisu saadaan komennolla Solve, jolle on annettava kaksi argumenttia: yhtälö ja tuntematon, so. symboli, jonka suhteen yhtälö halutaan ratkaista:

In[3]:=

ratkaisu = Solve[yhtalo, x]

Out[3]=

{{x -> 11/17}}

Yhtä hyvin yhtälö voidan suoraan kirjoittaa Solve-komennon argumentiksi:

In[4]:=

Solve[17 x - 2 == 9, x]

Out[4]=

{{x -> 11/17}}

Ratkaisu saadaan korvaussäännön muodossa. Muuttujalle x ei aseteta arvoa:

In[5]:=

x

Out[5]=

x

Mihin tahansa lausekkeeseen voidaan muuttujalle x sijoittaa saatu arvo korvausoperaattorin /. (eli ReplaceAll) avulla:

In[6]:=

x^2 /. First[ratkaisu]

Out[6]=

121/289

In[7]:=

yhtalo /. First[ratkaisu]

Out[7]=

True

Vastaavasti toisen asteen yhtälölle:

In[8]:=

yhtalo = a x^2 + b x + c == 0

Out[8]=

c + b x + a x^2 == 0

In[9]:=

ratkaisu = Solve[yhtalo, x]

Out[9]=

{{x -> (-b - (b^2 - 4 a c)^(1/2))/(2 a)}, {x -> (-b + (b^2 - 4 a c)^(1/2))/(2 a)}}

Tämän avulla voidaan laskea esimerkiksi juurten summa ja tulo:

In[10]:=

(x /. First[ratkaisu]) + (x /. Last[ratkaisu]) // Simplify

Out[10]=

-b/a

In[11]:=

(x /. ratkaisu[[1]]) * (x /. ratkaisu[[2]]) // Simplify

Out[11]=

c/a

Korvaussännön ratkaisu eri osiin on tässä viitattu ensin funktioilla First ja Last ja sitten indeksillä.

Vastaavaan tapaan voidaan käsitellä korkeampien asteiden yhtälöitä:

In[12]:=

Solve[x^3 - 6 x^2 - x + 18 == 0, x]

Out[12]=

{{x -> 2}, {x -> 2 - 13^(1/2)}, {x -> 2 + 13^(1/2)}}

Joskus tulos voi olla pitkähkö. Seuraavassa muodostetaan yleinen neljännen asteen yhtälön ratkaisukaava, joka talletetaan korvaussäännön muodossa nimelle rtk. Tämä sisältää itse asiassa neljä korvaussääntöä, koska neljännen asteen yhtälöllä on neljä juurta.

In[13]:=

poly = a x^4 + b x^3 + c x^2 + d x + e

Out[13]=

e + d x + c x^2 + b x^3 + a x^4

In[14]:=

rtk = Solve[poly == 0, x]

Out[14]=

{{x -> -b/(4 a) - 1/2 √ (b^2/(4 a^2) - (2 c)/(3 a) + (2^(1/3) (c^2 - 3 b d + 12 a e)) ... c^2 - 3 b d + 12 a e)^3 + (2 c^3 - 9 b c d + 27 a d^2 + 27 b^2 e - 72 a c e)^2)^(1/2))^(1/3)))))}}

Saadun ratkaisukaavan pätevyyttä voidaan testata muodostamalla ensin polynomi, jonka nollakohdat tiedetään, ja yrittämällä sitten ratkaista nämä edellä saadusta yleisestä kaavasta. Polynomi olkoon

In[15]:=

p = (x - 1) (x - 2) (x - 3) (x - 4) // Expand

Out[15]=

24 - 50 x + 35 x^2 - 10 x^3 + x^4

Sijoitetaan kertoimet edellä saatuun kaavaan ja yritetään sieventää:

In[16]:=

rtk /. {a -> 1, b -> -10, c -> 35, d -> -50, e -> 24} // Simplify

Out[16]=

{{x -> 5/2 - 1/2 (1/6 (20 - 26/(35 + 18 i 3^(1/2))^(1/3) - 2^(2/3) (70 + 36 i 3^(1/2))^(1/3 ... ^(1/2) + 1/2 (1/6 (10 + 26/(35 + 18 i 3^(1/2))^(1/3) + 2^(2/3) (70 + 36 i 3^(1/2))^(1/3)))^(1/2)}}

In[17]:=

rtk /. {a -> 1, b -> -10, c -> 35, d -> -50, e -> 24} // FullSimplify

Out[17]=

{{x -> 1}, {x -> 2}, {x -> 3}, {x -> 4}}

Aina ei kuitenkaan onnistu. Muodostetaan uusi neljännen asteen polynomi ja ratkaistaan se toisaalta suoraan Solve-komennolla, toisaalta sijoittamalla polynomin kertoimet ratkaisukaavaan.

In[18]:=

p = poly /. {a -> 1, b -> 0, c -> 0, d -> 0, e -> -1}

Out[18]=

-1 + x^4

In[19]:=

nollakohdat = Solve[p == 0, x]

Out[19]=

{{x -> -1}, {x -> -i}, {x -> i}, {x -> 1}}

Nämä ovat todellakin nollakohdat:

In[20]:=

p /. nollakohdat

Out[20]=

{0, 0, 0, 0}

Mutta:

In[21]:=

rtk /. {a -> 1, b -> 0, c -> 0, d -> 0, e -> -1}

                                         1 Power :: infy :  Infinite expression  -------  encountered.                                       Sqrt[0]

                                         1 Power :: infy :  Infinite expression  -------  encountered.                                       Sqrt[0]

                                         1 Power :: infy :  Infinite expression  -------  encountered.                                       Sqrt[0]

General :: stop :  Further output of  Power :: \" infy \"  will be suppressed during this calculation.

Out[21]=

{{x -> Indeterminate}, {x -> Indeterminate}, {x -> Indeterminate}, {x -> Indeterminate}}

Saatu ratkaisukaava ei siis olekaan aivan yleinen!

•Polynomiyhtälöt numeerisesti

Polynomiyhtälön ratkaiseminen ei aina ole suoraviivaista:

In[22]:=

p = x^7 + x + 1

Out[22]=

1 + x + x^7

In[23]:=

rtk = Solve[p == 0, x]

Out[23]=

{{x -> Root[1 + #1 + #1^7 &, 1]}, {x -> Root[1 + #1 + #1^7 &, 2]}, {x -> Root ... + #1^7 &, 5]}, {x -> Root[1 + #1 + #1^7 &, 6]}, {x -> Root[1 + #1 + #1^7 &, 7]}}

Nämä ovat kyllä polynomin nollakohdat, mutta niiden lausekkeet ovat hieman erikoiset. Taustalla on vakava periaatteellinen rajoitus: juurilausekkeiden (neliöjuuri, kuutiojuuri jne.) avulla ilmaistavat yleiset ratkaisukaavat ovat olemassa ainoastaan neljänteen asteeseen saakka. Korkeampien asteiden polynomeille voidaan käyttää eo. lausekkeiden tapaisia ilmauksia, jotka ovat oikeastaan vain tautologioita. Numeeriset arvot saadaan kuitenkin tällöinkin:

In[24]:=

rtk // N

Out[24]=

{{x -> -0.7965443541284571`}, {x -> -0.7052980879371502` - 0.6376237698062315` i}, {x -& ... 98083844899014`  - 0.5166768838399121` i}, {x -> 0.9798083844899014`  + 0.5166768838399121` i}}

Toisaalta numeeriset arvot voidaan laskea myös suoraan funktiolla NSolve, mikä saattaa olla oleellisestikin nopeampaa:

In[25]:=

NSolve[p == 0, x]

Out[25]=

{{x -> -0.7965443541284571`}, {x -> -0.7052980879371502` - 0.6376237698062315` i}, {x -& ... 98083844899014`  - 0.5166768838399121` i}, {x -> 0.9798083844899014`  + 0.5166768838399121` i}}

•Transkendenttiyhtälöt

Joissakin tapauksissa funktiota Solve voidaan käyttää myös muiden kuin polynomiyhtälöiden ratkaisuun:

In[26]:=

rtk = Solve[Sin[5 x] == Cos[3 x], x]

Solve :: ifun :  Inverse functions are being used by  Solve , so some solutions may not be found.

Out[26]=

{{x -> -(3 π)/4}, {x -> π/4}, {x -> -ArcCos[-(1/2 - 1/2 (1/2 - 1/(2 2^(1/2) ...  + 1/(2 2^(1/2)))^(1/2))^(1/2)]}, {x -> ArcCos[(1/2 + 1/2 (1/2 + 1/(2 2^(1/2)))^(1/2))^(1/2)]}}

In[27]:=

rtk // N

Out[27]=

{{x -> -2.356194490192345`}, {x -> 0.7853981633974483`}, {x -> -2.159844949342983`},  ... }, {x -> -1.3744467859455345`}, {x -> -2.9452431127404313`}, {x -> 0.19634954084936218`}}

Kaikkia juuria ei kuitenkaan saada.

Yleensä ns. transkendenttiyhtälöiden ratkaiseminen ei kuitenkaan onnistu:

In[28]:=

Solve[Log[x] == Tan[x], x]

Solve :: tdep :  The equations appear to involve the variables to be solved for in an essentially non-algebraic way.

Out[28]=

Solve[Log[x] == Tan[x], x]

Tällaisten yhtälöiden numeeriseen ratkaisemiseen on käytettävissä funktio FindRoot. Se käyttää Newtonin menetelmän tyyppistä (mutta tehokkaampaa) algoritmia yhtälön yhden ratkaisun löytämiseen. Algoritmille on annettava jonkinlainen etsittävän juuren alkuapproksimaatio:

In[29]:=

FindRoot[Log[x] == Tan[x], {x, 1}]

Out[29]=

{x -> 4.09546169232966`}

In[30]:=

FindRoot[Log[x] == Tan[x], {x, 6}]

Out[30]=

{x -> 7.390369571335114`}

In[31]:=

FindRoot[Log[x] == Tan[x], {x, 8}]

Out[31]=

{x -> 10.594836665204829`}

Menetelmä ei aina anna alkuapproksimaatiota lähintä juurta, kuten viimeinen esimerkeistä osoittaa.

Alkuarvojen etsimisessä voidaan käyttää graafisia esityksiä:

In[32]:=

Plot[{Log[x], Tan[x]}, {x, 0.01, 10}, PlotRange -> {-4, 4}]

[Graphics:HTMLFiles/luento2_69.gif]

Out[32]=

-Graphics -

•Yhtälöryhmät

Kaikkia edellä esitettyjä menetelmiä voidaan käyttää myös yhtälöryhmien ratkaisemiseen. Funktioissa Solve ja NSolve niin yhtälöt kuin tuntemattomatkin on tällöin annettava aaltosulkuihin suljettuina listoina:

In[33]:=

Solve[{a x + b y == s, c x + d y == t}, {x, y}]

Out[33]=

{{x -> -(-d s + b t)/(-b c + a d), y -> -(-c s + a t)/(b c - a d)}}

In[34]:=

Solve[{u^2 + v^2 == 1, v == u^2}, {u, v}]

Out[34]=

{{v -> -1/2 - 5^(1/2)/2, u -> -i (1/2 + 5^(1/2)/2)^(1/2)}, {v -> -1/2 - 5^(1/2)/2, u  ...  u -> -(-1/2 + 5^(1/2)/2)^(1/2)}, {v -> -1/2 + 5^(1/2)/2, u -> (-1/2 + 5^(1/2)/2)^(1/2)}}

In[35]:=

NSolve[{u^2 + v^2 == 1, v == u^2}, {u, v}]

Out[35]=

{{u -> 0.`  + 1.2720196495140685` i, v -> -1.6180339887498945`}, {u -> 0.`  - 1.27201 ... 574235`, v -> 0.6180339887498947`}, {u -> 0.7861513777574235`, v -> 0.6180339887498949`}}

Funktion FindRoot syntaksi on hieman erilainen:

In[36]:=

FindRoot[{u^2 + v^2 == 1, v == u^2}, {u, 1}, {v, 1}]

Out[36]=

{u -> 0.7861513777620805`, v -> 0.618033988749989`}


Converted by Mathematica  (May 8, 2003)