Periytyminen
Toinen oleellinen perustekniikka olio-ohjelmoinnissa koostumisen lisäksi on periytyminen. Periytymisellä tarkoitetaan, että luokka voi 'periä' toisen luokan kaikki ominaisuudet ja toiminnallisuudet (ja muokata niitä, sekä lisätä niihin omiaan). Tämä edistää koodin uudelleenkäyttöä, sillä tällä tekniikalla voimme hyödyntää jo olemassa olevaa koodia, mutta lisätä siihen tarvittavat uudet ominaisuudet tai muuttaa perittyjä osia.
Luokkahierarkiaa suunniteltaessa kannattaa yleiset, usean luokan tarvitsemat toiminnot ja ominaisuudet kerätä omiin luokkiinsa, joista sitten periytetään varsinaiset ohjelmassa käytettävät luokat. Periyttämisestä käytetään myös nimeä erikoistaminen (specialization) ja vastaavasti yhteisten metodien ja muuttujien siirtämistä omaan luokkaansa kutsutaan yleistämiseksi (generalization). Perimimisjärjestyksessä aikaisempaa luokkaa kutsutaan vanhemmaksi (parent) ja siitä perivää luokkaa lapseksi (child).
Periytyminen C++:ssa esitetään seuraavasti:unmigrated-wiki-markup
Code Block |
---|
class |
...
Perillinen : |
...
[määre |
...
] |
...
Vanhempi |
...
{
|
...
... } |
Wiki Markup |
---|
jossa \[määre\] on yleensä _public_ (joskus harvoin protected tai private). |
Anchor | ||||
---|---|---|---|---|
|
Esimerkki 1. Periytyminen.
Seuraavassa on esitetty ensin yleinen kantaluokka luokka CElain: Wiki Markup
elain.h:
Code Block |
---|
class CElain |
...
{
private: // saatavissa vain luokan funktioilla
|
...
char m_Kutsumanimi |
...
[100 |
...
];
|
...
float m_Massa; |
...
public: // voidaan käyttää luokan ulkopuolelta |
...
void AsetaNimi(char Nimi |
...
[ |
...
]);
|
...
char |
...
* KysyNimi( |
...
);
|
...
void AsetaMassa( |
...
float Massa); |
...
float KysyMassa(); }; |
Seuraavaksi on erikoistettu luokka, joka esittää eläintä kissa, siihen sisällytettään kaikki vain kissalle ominaiset piirteet.
class CKissa : public CElain
{
private: // saatavissa vain luokan funktioilla
public: // voidaan käyttää luokan ulkopuolelta
void Kehraa();
}
Luokan CKissa edustaja perii protected ja public-tyyppiset CElain-luokan metodit ja jäsenmuuttujat. Kts. näkyvyys.
siihen sisällytetään kaikki vain kissalle ominaiset piirteet.
kissa.h:
Code Block |
---|
#include "elain.h"
class CKissa : public CElain
{
private: // saatavissa vain luokan funktioilla
public: // voidaan käyttää luokan ulkopuolelta
void Kehraa();
};
|
Luokka CKissa perii kaikki CElain-luokan ominaisuudet ja toiminnallisuudet (=jäsenmuuttujat ja -funktiot). Mutta ainoastaan protected ja public-tyyppisiin päästään käsiksi CKissa-luokassa. Kts. näkyvyys.
Kuva 1. Esimerkki 1 UML-kielen luokkakaaviona esitettynäHuom! public-määre ennen perittävää luokkaa määrittelee perinnän näkyvyyden. Perinnässä voidaan käyttää myös private ja protected -määreitä, mutta niiden käyttö on erittäin harvinaista.
Vanhemman funktioiden ylikirjoitus perillisessä
Perityn luokan funktoita voidaan ylikirjoittaa perillisessä. Kutsuttaessa funktoita perillisen olion kautta suoritetaan perillisessä ylikirjoitettu funktio.
Anchor | ||||
---|---|---|---|---|
|
Esimerkki 2. Funktion ylikirjoitus perillisessä.
#include <iostream>
using namespace std;
class CElain
{
public: // voidaan käyttää luokan ulkopuolelta
void TulostaLuokanNimi()
{ cout << "Luokan nimi on CElain." << endl; }
};
class CLehma : public CElain
{
public:
void TulostaLuokanNimi()
{ cout << "Luokan nimi on CLehma." << endl; }
};
int main()
{
CLehma lehma;
lehma.TulostaLuokanNimi();
elain.h:
Code Block |
---|
#include <stdio.h>
class CElain
{
public: // voidaan käyttää luokan ulkopuolelta
void TulostaLuokanNimi()
{
printf("Luokan nimi on CElain.\n");
}
};
|
lehma.h:
Code Block |
---|
#include "elain.h"
class CLehma : public CElain
{
public:
void TulostaLuokanNimi()
{
printf("Luokan nimi on CLehma.\n");
}
};
|
main.cpp:
Code Block |
---|
#include "elain.h"
#include "lehma.h"
int main()
{
CLehma lehma;
lehma.TulostaLuokanNimi();
CElain elain;
elain.TulostaLuokanNimi();
}
|
Kuva 2. Esimerkin 2 luokkakaavio. CElain elain;
elain.TulostaLuokanNimi();
}
Testatkaa mitä main():ssa tehdyt funktiokutsut tulostavat.
Olion jäsenfunktioiden käyttö perillisestä
Vanhemman luokan jäsenfunktioita voidaan kutsua normaalisti, jos niitä ei ole ylikijoitettu.
Jos ne on ylikijoitettu, oletuksena kutsutaan aina perillisen versiota. Mieti seuraavaa tilannetta:
Code Block |
---|
#include "elain.h"
class CLehma : public CElain
{
public:
void TulostaLuokanNimi()
{
TulostaLuokanNimi();
printf("Luokan nimi on CLehma.\n");
}
};
|
Tämä on loputon rekursio, eli funktio kutsuu itseään loputtomasti, sillä siinä ei ole lopetusehtoa.
Jos taas tarkoituksena oli kutsua vanhemman (CElain) TulostaLuokanNimi()-funktiota täytyy se määritellä eksplisiittisesti:
Code Block |
---|
#include "elain.h"
class CLehma : public CElain
{
public:
void TulostaLuokanNimi()
{
CElain::TulostaLuokanNimi();
printf("Luokan nimi on CLehma.\n");
}
};
|
Tämä koodi toimii halutulla tavalla, eli CLehma-luokasta luodun olion TulostaLuokanNimi()-funktio kutsuu ensin vanhempansa, eli CElain-luokan funktiota TulostaLuokanNimi(), jonka jälkeen se tulostaa ruudulle oman tekstinsä.