Python (8.) - OOP v Pythonu

Po minulém teoretickém díle se dnes podíváme, jak vypadá OOP v Pythonu.

3.2.2005 09:00 | Aleš Hakl | přečteno 25414×

Novou třídu v Pythonu definujeme prostřednictvím klíčového slova class. Za ním uvedeme jméno nové třídy a dvojtečkou otevřeme blok, v tomto bloku můžeme definovat metody naší nové třídy, pokud žádné metody definovat nechceme, použijeme klíčové slovo pass.

class PrazdnaTrida:
  pass

Metody definujeme stejně jako funkce klíčovým slovem def, jejich první parametr (obvykle nazývaný self) se při volání neuvádí a obsahuje instanci, pro kterou je metoda volána. Na rozdíl od jiných jazyků musíme při práci s instancí self vždy tento identifikátor uvést.

class MojeTrida:
  def moje_metoda(self):
    print self,": moje_metoda"

  def dalsi_metoda(self,a,b):
    print "Self=",self,"a=",a,"b=",b

Takto vytvořená třída se chová podobně jako funkce, můžeme ji zavolat, čímž získáme její instanci.

>>> a = MojeTrida()
>>> a
<__main__.MojeTrida instance at 0x403119cc>
>>> a.moje_metoda()
<__main__.MojeTrida instance at 0x403119cc> : moje_metoda

Metody, jejichž názvy začínají i končí dvěma podtržítky, mají speciální význam. Nejzajímavější je jistě metoda __init__, ta je volána (pokud existuje) při vytváření instancí dané třídy s parametry, které jsou při této operaci použity. Obdobným způsobem je před výmazem instance volána metoda __del__, ovšem její využívání přináší různé problémy, a proto se příliš nepoužívá.

Metoda __init__ je vhodné místo k vytvoření atributů instance (proměnných specifických pro danou instanci):

class Bod:
  def __init__(self,x=0,y=0):
    self.x = x
    self.y = y
  def posun(dx,dy):
    self.x = self.x + dx
    self.y = self.y + dy

Všimněte si že jsem parametrům x i y přiřadil výchozí hodnotu, obecně je možné v parametrech libovolné metody použít libovolnou možnost uvedenou v předminulém díle o funkcích, pouze je nutné počítat s tím, že v prvním argumentu bude vždy instance, se kterou budeme pracovat.

Instanci naší třídy Bod tedy můžeme vytvořit například těmito způsoby:

a = Bod()
b = Bod(1,1)
c = Bod(x=1,y=1)

Z příkladu doufám jasně vyplývá jakým způsobem s atributy instance manipulujeme, není to ovšem jediná možnost, instance jsou totiž interně implementovány jako slovníky, ke kterým můžeme přímo přistupovat pomocí atributu __dict__. Předchozí příklad tedy také můžeme zapsat jako:

class Bod:
  def __init__(self,x=0,y=0):
    self.__dict__["x"] = x
    self.__dict__["y"] = y
  def posun(dx,dy):
    self.__dict__["x"] = self.__dict__["x"] + dx
    self.__dict__["y"] = self.__dict__["y"] + dy

Ve skutečnosti je mezi oběma variantami jistý rozdíl, povíme si o něm v některém z dalších dílů spolu s takzvanými speciálními metodami.

Je také možné definovat atributy tříd, neboli proměnné spojené se třídou a nikoli jednotlivými instancemi. Provedeme to tak, že uvnitř bloku, ve kterém definujeme metody, nadefinujeme proměnnou:

class SerioveCislo:
  cislo = 0
  def __init__(self):
    SerioveCislo.cislo = SerioveCislo.cislo + 1
    self.cislo = SerioveCislo.cislo

V příkladu opět vidíme, jak k takové proměnné přistupujeme, druhý řádek metody __init__ ukazuje, že pokud vytvoříme atribut instance se stejným názvem jako atribut třídy, zastíní atribut instance atribut třídy. V našem případě budou mít tedy všechny instance výše uvedené třídy v atributu cislo svoje unikátní číslo.

>>> a=SerioveCislo()

>>> b=SerioveCislo()
>>> c=SerioveCislo()
>>> a.cislo
1
>>> c.cislo
3
>>> b.cislo
2

Při definování třídy můžeme do závorky za její jméno uvést seznam předků, v případě, že vznikne kolize jmen, má přednost dříve uvedená třída.

>>> class OdvozenaTrida(MojeTrida):
...     def nova_metoda(self,a,b):
...             MojeTrida.moje_metoda(self)
...             MojeTrida.dalsi_metoda(self,a,b)
... 
>>> b = OdvozenaTrida()
>>> b.nova_metoda(1,2)
<__main__.OdvozenaTrida instance at 0x40311bcc> : moje_metoda
Self= <__main__.OdvozenaTrida instance at 0x40311bcc> a= 1 b= 2

Z předchozího příkladu je také patrné, jak je možné volat metodu konkrétní třídy. Dodejme, že to je možné odkudkoli, nejen ze tříd od dané třídy odvozené. Taktéž jako parametr self můžeme předat celkem cokoli (většinou to ovšem povede k chybě s tím, že self postrádá nějaký atribut, se kterým daná metoda potřebuje pracovat).

Metody jsou ve skutečnosti také atributy, v případě tříd jsou to atributy s datovým typem funkce a v případě instancí se speciálním typem vázaná metoda (bound method). Vázaná metoda je funkce, jejíž první argument je vždy odpovídající instance. Z tohoto faktu vyplývá několik zajímavých důsledků:

Pokud vás zaráží, že jsem vůbec nezmínil tzv. řízení přístupu, je to zcela správně, Python totiž podobný koncept vůbec nezavádí, s výjimkou možnosti zapsat před název metody nebo atributu dvě podtržítka, v tom případě Python daný název převede na _JménoTřídy__původní_jméno, například __nyaa ve třídě Neko převede na _Neko__nyaa.

Existuje funkce isinstance(něco, třída), vracející True v případě, že něco je instancí třídy třída (nebo některé třídy od ní odvozené), a issubclass(podtřída, třída) vrátí True tedy, je-li podtřída odvozena od třídy třída. Osobně pro tyto třídy nevidím velké využití, síla objektového modelu jazyka Python tkví právě v tom, že můžeme vzájemně zaměňovat instance na základě toho, jaké mají rozhraní a nikoli toho, od čeho je jejich třída odvozená, a proto mi přijde zbytečné tuto sílu omezovat nějakými explicitními kontrolami.

V příštím díle se podíváme na takzvané moduly, čímž ukončíme nudnou pouť po nutných základech jazyka Python a přesuneme se k jistě mnohem zajímavějšímu tématu - k různým funkcím, modulům a třídám, které nám Python nabízí.

Online verze článku: http://www.linuxsoft.cz/article.php?id_article=671