V tomto díle si vysvětlíme, co je to speciální metoda, a naprogramujeme vlastní třídu, která se bude chovat jako seznam.
Speciální metody jsou vámi definované metody ve vašich třídách. Tyto metody nejsou volány jejich jménem, ale jsou zavolány, když Python zaregistruje požadavek vznesený na třídu.
Ukážeme si to na konstruktoru a destruktoru. Konstruktor je zavolán při vzniku třídy a destruktor při zániku:
>>> class Ukazka:
def __init__(self):
print "Zavolan konstruktor"
def __del__(self):
print "Zavolan destruktor"
>>> trida=Ukazka()
Zavolan konstruktor
>>> del trida
Zavolan destruktor
Speciálních metod je celá řada, ale protože je tento díl věnován spíše začínajícím programátorům, uvedu pouze některé:
Jméno metody | Popis metody |
__init__ | Volána vždy při vytváření třídy. První parametr je instance. Tato funkce musí vždycky vracet None. |
__del__ | Tato metoda je volána při zániku třídy. |
__str__ | Tato metoda je zavolána, když zavoláte str(trida). |
__len__ | Voláno, když Python narazí na len(objekt). |
__getitem__ | Voláno, když Python vyhodnocuje objekt[index]. |
__setitem__ | Voláno, když Python vyhodnocuje objekt[index]=vyraz. |
__delitem__ | Tato metoda se volá, když Python narazí na del objekt[index]. |
__setslice__ | Voláno, když je vyhodnocováno objekt[i:j]=vyraz. |
__add__ | Voláno, když je vyhodnocováno vysledek=objekt+vyraz. |
__sub__ | Odpovídá metodě __add__ s tím rozdílem, že je volána, když je vyhodnocováno vysledek=objekt-vyraz. |
__mul__ | Odpovídá metodě __add__ s tím rozdílem, že je volána, když je vyhodnocováno vysledek=objekt*vyraz. |
__div__ | Odpovídá metodě __add__ s tím rozdílem, že je volána, když je vyhodnocováno vysledek=objekt/vyraz. |
Pozor, metoda __add__ funguje, pouze když je objekt na levé straně sčítání, tedy vysledek=objekt+vyraz. Aby to fungovalo i naopak, je nutné definovat další speciální metodu __radd__.
Příklad
Využití některých metod si ukážeme na vytvoření třídy, která se bude chovat podobně jako seznam.
# -*- coding: cp1250 -*-
class NasSeznam:
def __init__(self, seznam=[]):
self.seznam=seznam
if type(self.seznam) != list:
raise TypeError, "Základem musí byt seznam."
def pridej(self, prvek):
self.seznam.append(prvek)
def __getitem__(self, index):
return self.seznam[index]
def __setitem__(self,index,prvek):
self.seznam[index]=prvek
def __add__(self, seznam):
if type(seznam) != list:
raise TypeError, "Přičítat se mohou pouze seznamy."
return NasSeznam(self.seznam+seznam)
def __len__(self):
return len(self.seznam)
def __delitem__(self, index):
del self.seznam[index]
def __setslice__(self,zacatek, konec, nahrada):
self.seznam[zacatek:konec]=nahrada
def __str__(self):
return str(self.seznam)
A tady je malá ukázka, jak náš seznam funguje:
>>> katalog=NasSeznam([1,2,3])
>>> print katalog
[1, 2, 3]
>>> katalog.pridej(4)
>>> print katalog
[1, 2, 3, 4]
>>> katalog=katalog+[5,6]
>>> print katalog
[1, 2, 3, 4, 5, 6]
>>> print katalog[0:1]
[1]
>>> katalog[0:2]="nic"
>>> print katalog
['n', 'i', 'c', 3, 4, 5, 6]
>>> del katalog[0]
>>> print katalog
['i', 'c', 3, 4, 5, 6]
>>> katalog=katalog+5#nefunguje, protože k seznamům mohu přičítat pouze seznamy.
Traceback (most recent call last):
File ">pyshell#18>", line 1, in -toplevel-
katalog=katalog+5
File "C:\Documents and Settings\Blu\Plocha\specialni_metody.py", line 13, in __add__
if type(seznam) != list:raise TypeError, "Základem musí být seznam."
TypeError: Přičítat se mohou pouze seznamy.
V dalším dílu si řekneme něco o dědičnosti.