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:
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:
elain.h:
Code Block |
---|
class CElain { private: // saatavissa vain luokan funktioilla char m_Kutsumanimi\[100\]; int float m_Massa; public: // voidaan käyttää luokan ulkopuolelta void AsetaNimi(char Nimi\[\]); char\* KysyNimi(char Nimi\[\]); void AsetaMassa(intfloat Massa); intfloat KysyMassa(); }; |
Seuraavaksi on erikoistettu luokka, joka esittää eläintä kissa, siihen sisällytettään 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(); }; |
Luokan Luokka CKissa edustaja perii perii kaikki CElain-luokan ominaisuudet ja toiminnallisuudet (=jäsenmuuttujat ja -funktiot). Mutta ainoastaan protected ja public-tyyppiset CElain-luokan metodit ja jäsenmuuttujattyyppisiin 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ä.