ARCHIV |
|||||
Software (10844)
Distribuce (131)
Skripty (697)
Menu
Diskuze
Informace
|
Perl (60) - OOP - dědičnostDědičnost umožňuje vytvářet mezi třídami vztahy, díky nimž se pak celý systém tříd stává daleko přehlednějším. Biologicky řečeno, dědičnost (inheritance) je nějaký děj, ve kterém organizmus získá vlastnosti svého rodiče. Když zanedbáme evoluci (chyby dědičnosti) a druhy organizmů budeme považovat za dané, pak vztahy mezi druhy vytvářejí představu dědičnosti v objektově orientovaném programování. Dědičnost tedy znamená, že skupina tříd je organizována ve stromu na základě vzájemné podobnosti - podobnosti v tom smyslu, že jedna třída je nějakým zobecněním druhé. Pak první třídu nazveme potomkem druhou rodičem. V takovém případě se třída potomka nepíše znovu, ale je prohlášena za dědící od rodiče a získá automaticky jeho vlastnosti, ke kterým si pak může přidat další. Máme-li podobné třídy, které ale nemají vztah dědičnosti, pak je možné, že společně dědí od jiné třídy. Třeba i třídy, která nemusí mít smysl, tj. u které nemá smysl hovořit o instancích. V takovém případě hovoříme o tzv. abstraktní třídě. Význam abstraktních tříd uvidíme na následujícím příkladu. Příklad - vlastnosti dědičnosti, abstraktní třídyPojďme se ještě na chvíli vrátit k biologii. Pokusíme se srovnat dědičnost s biologickým členěním organizmů a ukázat si na větším teoretickém příkladě, jak vlastně dědičnosti využít. Organizmus obecně bude popisovat nějaká třída Organizmus. Její definice musí být natolik obecná, aby vyhovovala naprosto všem organizmům. Dejme tomu, že všechny organizmy musí dýchat a definujme metodu dýchej(). Každý organizmus je v určitém okamžiku na nějaké pozici souřadného systému a tak má svůj atribut pozice. Třída Organizmus je abstraktní, protože nemá smysl vytvářet objekty typu Organizmus. První specializací třídy Organizmus je dělení na živočichy, rostliny, houby apod. Pro každou tuto kategorii bude existovat jedna třída. O každé z těchto tříd (Živočich, Rostlina, Houba) můžeme říci, že je organizmem. Mají vlastnosti Organizmu (dýchej() atd.) a k tomu si přidávají některé svoje další. Například v třídě Rostlina bude implementována metoda fotosyntéza() nebo Živočich se bude moci přijímat potravu metodou přijmi_potravu(). Každý živočich bude mít orgán na vnímaní světla a tedy i definován atribut ostrost_zraku. Toto všechno jsou také abstraktní třídy. A takto můžeme jít stále dál. Z třídy živočich dědí další třídy jako Pták, Savec nebo Plaz. Třída Pták sdílí všechny vlastnosti třídy Živočich (přijmi_potravu(), dýchej() atd.) a přidává si jiné, například metodu leť(x, y, z). Udělejme poslední krok, abychom se dostali z oblasti abstraktních tříd. Z třídy pták dědí třídy Kos a Vrabec. Nyní můžeme vytvářet objekty, které jsou instancemi těchto tříd. A právě zde uvidíme vlastnosti dědičnosti. Objekt $vrabec1 i objekt $lípa1 mohou dýchat. Přesto jsme nemuseli metodu dýchej() od základů definovat vícekrát. Dýchání je obecná funkce třídy Organizmus, z níž oba naše objekty dědí. Oba objekty také mají v souřadném systému nějakou pozici. Ale ostrost_zraku $lípa1 už nemá a naopak $vrabec1 neumí fotosyntézu. Vícenásobná dědičnostDědičnost lze rozdělit na jednoduchou a vícenásobnou. Liší se v tom, od kolika tříd je děděno. Pokud naše třída dědí z jedné třídy, jde o jednoduchou dědičnost. Pokud jich je více nazývá se vícenásobná. Opět lze hledat analogie v biologii, tentokrát uvnitř jednoho druhu. Některé druhy organizmů mohou mít více než jednoho rodiče (tedy obvykle dva). Perl podporuje oba typy dědičnosti, ale u všech objektových jazyků tomu tak není. Některé podporují pouze jednoduchou. Vícenásobná dědičnost má své zastánce i odpůrce. Pravdou je, že se s ní člověk příliš často nesetká, ale na druhou stranu ji, pokud je potřeba, nelze jen tak něčím nahradit. Dědičnost v PerluFormální označení třídy jako potomka jiné třídy obnáší jedinou věc. Rodičovskou třída je třeba v odvozené třídě označit pomocí pole @ISA (anglické "is a" vyjadřuje vztah náležení). Přesněji řečeno, toto pole obsahuje jména všech tříd, ze kterých aktuální třída dědí, což je obvykle (tedy v případě jednoduché dědičnosti) jedna třída. Rodičovská třída může obsahovat pole @ISA také (jinak řečeno může být zároveň potomkem ještě obecnější třídy). Tímto způsobem pak můžeme tvořit různé hierarchické vztahy. Ovšem většinou budeme chtít provádět i jiné věci. Nezbytné bude zejména předefinovávání metod předků. Pojďme se podívat na další syntaktické jevy u dědičnosti a v dalším díle si ukážeme jejich užití. Třída SUPERSUPER je speciální třídou, která zastupuje předka. Zavoláním třídy SUPER se rekurzivně spouští hledání metody v rodičovských třídách. SUPER::new má tedy v případě jednoduché dědičnosti ve výsledku podobný smysl jako Předek::new. SUPER::new použijeme, pokud nechceme použít konkrétní třídu, ale nechat si vybrat podle hierarchie definované v @ISA. Využití SUPER je poměrně široké. Často se s ní setkáme v konstruktorech potomků. Voláním metody SUPER::new totiž vyvoláváme konstruktor předka, který vytvoří objekt, a ten pak pomocí new upravíme. Předefinování metodyMetoda třídy může upravovat metody tříd, ze kterých tato třída dědí. Ukažme si nesmyslnou, ale jednoduchou ukázku. Máme třídu A, ve které je definována metoda tiskni_atribut. Ta tiskne aktuální hodnotu konkrétního atributu. Třída B, která dědí z A bude chtít metodu tiskni_atribut používat, ale jiným způsobem. Tisknout se bude pouze v případě, že atribut je posloupností číslic. V opačném případě vytiskneme chybovou hlášku. Takto bude vypadat metoda B::tiskni_atribut.
V příštím díle se předefinováváním budeme zabývat podrobněji. Názvy dědících třídDalší věcí, která stojí za pozornost, je pojmenování třídy za pomoci čtyřtečky ::. Tento symbol má dva významy. Už z názvu třídy je z takového zápisu patrný příbuzenský vztah. Čtyřtečka si také vynucuje přehlednější adresářovou strukturu. Modul Math::Functions se bude ve skutečnosti hledat pod názvem Math/Functions.pm. Univerzální třída a její metodyZ třídy s názvem UNIVERSAL dědí všechny třídy mimo UNIVERSAL. UNIVERSAL je jediná třída, která nemá předka. To má několik zajímavých důsledků - například to, že každá námi definovaná třída bude mít alespoň jednoho předka. UNIVERSAL obsahuje tři metody. Ihned z definice třídy UNIVERSAL plyne, že tyto metody zdědí každá naše třída. Pojďme se na ně tedy podívat. Metoda canTato metoda slouží ke zjištění, zda lze nad daným objektem volat danou metodu. Jako argument předáme metodě can jméno metody. Pokud taková metoda skutečné existuje (lze ji volat nad daným objektem bez použití AUTOLOAD), can vrací pravdivou hodnotu.
Metoda isaPokud je argument platným názvem třídy, ze které třída, nad níž je metoda isa volána, dědí, je vrácena pravdivá hodnota.
Metoda VERSIONVe všech modulech existuje speciální proměnná $VERSION. Do ní můžeme přiřadit jakékoliv desítkové číslo, které označuje verzi modulu.
our $VERSION = 3.3;
Uživatel modulu si nyní může stanovit jeho nejnižší verzi, která je pro něj únosná. Pokud bude chtít verzi minimálně 2.3, zadá pro import tento příkaz.
use Modul 2.3;
To, že předchozí řádek funguje, obstarává právě metoda VERSION. Je-li zavolána, ukončí program, pokud je verze zadaná jako argument větší než dostupná verze modulu. Systém hledání volané metodyPojďme si stanovit přesná pravidla, jakými se řídí hledání metody po jejím zavolání. Budeme uvažovat obecný případ, tedy vícenásobnou dědičnost. Ze všeho nejdříve je hledána metoda v třídě, jehož je náš objekt instancí. Pokud taková třída neexistuje, prohledávají se rekurzivně všechny rodičovské třídy, podle toho, v jakém pořadí jsou v poli @ISA uvedeny. Třída UNIVERSAL je implicitně uvedena vždy naposled. Pokud není metoda nalezena, hledá se stejným způsobem speciální metoda AUTOLOAD. Pokud žádná vyhovující metoda není nalezena, dojde k chybě. Ukažme si pro lepší představu konkrétní příklad. definujeme si třídu Smrk.
Nyní nad třídou Smrk nebo její instancí zavoláme metodu metoda. Smrk::metoda tedy bude hledána postupně tak, jak je uvedeno v následujícím seznamu, dokud některá z nich nebude nalezena.
Související články
Předchozí Celou kategorii (seriál) Další
Perl (1) - Dávka teorie na úvod
Perl (2) - Úvod do syntaxe Perl (3) - Proměnné Perl (4) - Čísla a řetězce Perl (5) - Podmínky Perl (6) - Pravdivostní výrazy Perl (7) - Vstup poprvé Perl (8) - Některé základní vestavěné funkce Perl (9) - Cykly Perl (10) - Další řídící struktury Perl (11) - Pole - úvod Perl (12) - Pole - základní operace Perl (13) - Hashe Perl (14) - Další nástroje pro seznamy Perl (15) - Výchozí proměnná, heredoc, symbolické odkazy Perl (16) - Regulární výrazy - začínáme Perl (17) - Regulární výrazy - kotvy Perl (18) - Regulární výrazy - množiny znaků Perl (19) - Regulární výrazy - opakování a kvantifikátory Perl (20) - Regulární výrazy - magické závorky Perl (21) - Regulární výrazy - nahrazování Perl (22) - Regulární výrazy - přepínače Perl (23) - Regulární výrazy - rozšířené vzory Perl (24) - Regulární výrazy - příklady Perl (25) - Regulární výrazy - závěr Perl (26) - Podprogramy Perl (27) - Prototypy Perl (28) - Rozsahy platnosti proměnných Perl (29) - Úvod k práci se soubory Perl (30) - Práce se soubory Perl (31) - Testování souborů Perl (32) - Jiné typy souborů Perl (33) - Formátování výstupu - printf Perl (34) - Formátování výstupu - formáty Perl (35) - Vestavěný debugger Perl (36) - Grafické debuggery Perl (37) - Začínáme s moduly Perl (38) - Rozhraní modulu Perl (39) - Pragma Perl (40) - Dodatky k modulům Perl (41) - CPAN Perl (42) - Argumenty příkazového řádku Perl (43) - Přepínače Perl (44) - Dlouhé přepínače Perl (45) - Odkazy Perl (46) - Užití odkazů a anonymní data Perl (47) - Složitější datové struktury Perl (48) - Libovolně složité datové struktury Perl (49) - Tabulky symbolů a typegloby Perl (50) - Uzávěry a iterátory Perl (51) - Signály Perl (52) - Externí příkazy Perl (53) - Režim nakažení Perl (54) - Fork Perl (55) - Eval Perl (56) - Volby příkazu perl Perl (57) - Jednořádkové skripty Perl (58) - OOP - úvod Perl (59) - OOP - typické použití Perl (61) - OOP - přínos a užití dědičnosti Perl (62) - OOP - přetěžování Perl (63) - OOP - závěr Perl (64) - Projekt - čtečka sportovních výsledků Perl (65) - Projekt - získání dat Perl (66) - Projekt - výběr zápasů a podrobnosti Perl (67) - Projekt - dokončujeme modul Perl (68) - Projekt - zobrazení zápasů Perl (69) - Projekt - online přenos Perl (70) - Plain Old Documentation Perl (71) - Navazování proměnných Perl (72) - Navazování složitějších datových typů Perl (73) - DBM Perl (74) - Sockety Perl (75) - Obsluha více klientů Perl (76) - Síťová hra v kostky Perl (77) - Služby internetu Perl (78) - Databáze - úvod Perl (79) - Databáze - manipulace s daty Perl (80) - Databáze - závěrečné poznámky Perl (81) - CGI - příprava webového serveru Perl (82) - CGI - první skripty Perl (83) - CGI - získávání dat od uživatele Perl (84) - CGI - usnadnění tvorby skriptů pomocí modulu CGI Perl (85) - CGI - generování dokumentu modulem CGI Perl (86) - CGI - cookies Perl (87) - CGI - příklad aplikace Perl (88) - CGI - závěr Perl (89) - Mason - snadné psaní webů Perl (90) - Mason - speciální bloky Perl (91) - Mason - handlery Perl (92) - Mason - závěr Perl (93) - Catalyst - MVC framework pro Perl Perl (94) - Catalyst - základy pro psaní aplikace Perl (95) - Catalyst - šablony Perl (96) - Catalyst - spolupráce s databází Perl (97) - Curses - tvorba textových uživatelských rozhraní Perl (98) - Curses - pozicování a okna Perl (99) - Curses - měření rychlosti psaní Perl (100) - Curses - použití hotových widgetů Perl (101) - Curses - jednoduchý textový editor Perl (102) - Rozšiřování Perlu pomocí XS Perl (103) - Rozšiřování Perlu pomocí SWIG Perl (104) - Testování rychlosti Perl (105) - Testování programových jednotek Perl (106) - Debugování pomocí komentářů Perl (107) - Moose - moderní objektový systém Perl (108) - Moose - základní vlastnosti Perl (109) - Moose - role Perl (110) - Moose - meta API Perl (111) - Pokročilá práce se seznamy Perl (112) - Práce s PDF Perl (113) - Práce s archivy Perl (114) - Tk - úvod Perl (115) - Tk - umísťování widgetů Perl (116) - Tk - základní widgety Perl (117) - Tk - některé pokročilejší widgety Perl (118) - Tk - čas a události Perl (119) - Tk - CD man Perl (120) - Wx - základní práce s widgety Perl (121) - Wx - události Perl (122) - Gtk2 - úvod Perl (123) - Gtk2 - základní práce s obrázky Perl (124) - Gtk2 - události a čas Perl (125) - Gtk2 - vlastní widgety Perl (126) - Gtk2 - textové okno a práce s pozicemi Perl (127) - Gtk2 - hierarchické seznamy Perl (128) - Gtk2 - dialogy Perl (129) - Gtk2 - skládání widgetů Perl (130) - Gtk2 - menu a toolbary Perl (131) - Gtk2 - transparentní okna, tray ikona, výběr souborů Perl (132) - Gtk2 - drag&drop, druid Perl (133) - Gtk2 - úpravy vzhledu aplikací pomocí rc Perl (134) - Gtk2 - Glade Interface Designer Perl (135) - XML - čtení a zápis Perl (136) - XML - DOM a SAX přístupy Perl (137) - Vlákna Perl (138) - Memoizace - cachování podprogramů Perl (139) - Profilling - efektivní odhalování pomalých míst v programu Perl (140) - Profilling - píšeme si vlastní profiler / debugger Perl (141) - Formátování kódu, deparsování, perltidy Perl (142) - Způsoby konfigurování Perl (143) - Struktura datových typů, správa paměti Perl (144) - POE - událostmi řízené programování Perl (145) - POE - aplikace typu klient-server Perl (146) - Perl 6 - jazyk budoucnosti Perl (147) - Perl 6 - regulární výrazy, nové operátory Perl (148) - Perl Culture Perl (149) - Závěr Pozvánka na Český Perl Workshop Perl 5.22.0 a vše okolo Perl 5.24.0 a vše okolo Předchozí Celou kategorii (seriál) Další
|
Vyhledávání software
Vyhledávání článků
28.11.2018 23:56 /František Kučera 12.11.2018 21:28 /Redakce Linuxsoft.cz 6.11.2018 2:04 /František Kučera 4.10.2018 21:30 /Ondřej Čečák 18.9.2018 23:30 /František Kučera 9.9.2018 14:15 /Redakce Linuxsoft.cz 12.8.2018 16:58 /František Kučera 16.7.2018 1:05 /František Kučera
Poslední diskuze
31.7.2023 14:13 /
Linda Graham 30.11.2022 9:32 /
Kyle McDermott 13.12.2018 10:57 /
Jan Mareš 2.12.2018 23:56 /
František Kučera 5.10.2018 17:12 /
Jakub Kuljovsky | |||
ISSN 1801-3805 | Provozovatel: Pavel Kysilka, IČ: 72868490 (2003-2024) | mail at linuxsoft dot cz | Design: www.megadesign.cz | Textová verze |