Ero sivun ”Muistivuoto” versioiden välillä

Wikipediasta
Siirry navigaatioon Siirry hakuun
[katsottu versio][katsottu versio]
Poistettu sisältö Lisätty sisältö
lähteetön
Ipr1 (keskustelu | muokkaukset)
Ei muokkausyhteenvetoa
Rivi 29: Rivi 29:
}
}
</source>
</source>

[[Luokka:Ohjelmointi]]
[[Luokka:Ohjelmointivirheet]]

Versio 31. tammikuuta 2020 kello 00.55

Muistivuoto on tyypillinen ohjelmointivirhe, jossa tietokoneohjelma ei vapauta varaamaansa muistia kun sitä ei enää tarvita. Kun muistivuotoa sisältävää ohjelman osaa suoritetaan riittävän monta kertaa, voi vapaa muisti loppua, mikä aiheuttaa ohjelmiston kaatumisen. Joissakin ohjelmointikielissä, kuten Javassa, on automaattinen roskienkerääjä, joka pyrkii vapauttamaan ohjelmiston varaamia resursseja ohjelman uusiokäyttöön. Tällaista automaattista roskienkerääjää ei ole kuitenkaan esimerkiksi C:ssä ja C++:ssa. Se voidaan toteuttaa olemassa olevilla kirjastoilla tai toteuttaa itse ohjelman päättyessä kirjoittamalla kaikkien muuttujien päälle nollia. Muistin yli- ja alivuototilanteet on myös hyvä ottaa huomioon.

Ongelman välttäminen

Muistivuodon havaitsemista varten voidaan tehdä testiohjelma, joka suorittaa jotakin funktiota useita kertoja peräkkäin. Jos ohjelma kaatuu koko ajan samassa suoritusvaiheessa, saatetaan havaita muistivuotoa aiheuttava koodin ongelmakohta. Olemassa olevan koodin voi myös tutkia sekä kaupallisilla, että ilmaisilla työkaluilla, esimerkiksi IBM Rational Purify, BoundsChecker, Valgrind, Insure++ ja memwatch ovat joitakin suosituimpia C- ja C++-kielen muistivuodon virheidenjäljitysohjelmia.

Muistin yli- ja alivuototilanne

Muistin ylivuototilanteessa kirjoitetaan kirjoitettavaksi tarkoitetun alueen yli, esimerkiksi kirjoittamalla kymmenen tavua viiden tavun alueelle. Muistin alivuototilanteessa sijoitetaan muuttuvaksi määritellyn alueen keskelle merkki alueen päättymisestä, minkä on tarkoitus ilmoittaa konekielisen ohjeistuksen alkamisesta.

Yksinkertainen C-kielinen esimerkki

Tämä ohjelma havainnollistaa muistivuotoa.

 int main(void)
 {
    char* string1 = malloc(50);
    char* string2 = malloc(50);
    scanf("%s", string2);
    string1 = string2; 
              /* Tässä string1:n varaama muisti kadotetaan.
              *  Se ei osoita mihinkään muistialueeseen,
              *  joten sitä ei voida eksplisiittisesti vapauttaa.
              */
    free(string2); /* Tämä toimii. */
    free(string1); /* Muistivuoto! Yritetään vapauttaa muistia 
                      joka on jo kertaalleen vapautettu, 
                      ja string1:lle varattu muisti jää vapauttamatta.*/
    return 0;
 
 }