Fotografický magazín "iZIN IDIF" každý týden ve Vašem e-mailu.
Co nového ve světě fotografie!
Zadejte Vaši e-mailovou adresu:
Kamarád fotí rád?
Přihlas ho k odběru fotomagazínu!
Zadejte e-mailovou adresu kamaráda:
Informace
Nastal čas na kakao - Málem bychom zapomněli: NSAutoreleasePool
30. března 2005, 00.00 | Pokud bychom chtěli dodržet pořadí, v němž jsme se o třídách Foundation Kitu zmínili v úvodním přehledovém článku (a to bychom měli, aby bylo snadné se v nich orientovat), neměli jsme začínat kontejnerovými třídami NS(Mutable)Array, ale třídou NSAutoreleasePool, v níž (spolu s některými službami třídy NSObject) je skryto celé kouzlo poloautomatického garbage collectoru firmy NeXT – pardon, firmy Apple ☺.
Pokud bychom chtěli dodržet pořadí, v němž jsme se o třídách Foundation Kitu zmínili v úvodním přehledovém článku (a to bychom měli, aby bylo snadné se v nich orientovat), neměli jsme začínat kontejnerovými třídami NS(Mutable)Array, ale třídou NSAutoreleasePool, v níž (spolu s některými službami třídy NSObject) je skryto celé kouzlo poloautomatického garbage collectoru firmy NeXT – pardon, firmy Apple ☺.
Ovšem, pro pochopení příkladů kódu, jež zde uvádíme, je vhodné, abychom už měli za sebou alespoň povídání o třídě NSValue, a v navazujícím příštím článku o výjimkách bude dobře rozumět distribuovaným objektům. Proto zmíněné "přehlédnutí" napravíme až dnes: vysvětlíme si, jaké je postavení třídy NSAutoreleasePool ve Foundation Kitu, a ukážeme si, jakým způsobem podporuje automatický zánik objektů "když je to zapotřebí".
Základy
Základy správy paměti v Cocoa jsme si už vysvětlili dávno, hned na začátku našeho seriálu v článku, věnovanému vzniku a zániku objektů. Nyní proto zopakujeme jen ta nejzákladnější pravidla (jež si dnes upřesníme a podrobněji vysvětlíme):
- objekt, který explicitně vytvoříme kombinací zpráv alloc/init..., pomocí zastaralé zprávy new, nebo pomocí zpráv copy a mutableCopy, není spravován garbage collectorem: musíme jej proto explicitně uvolnit pomocí zprávy autorelease nebo release;
- získáme-li objekt jakkoli jinak, je spravován garbage collectorem, a zanikne (nějaký čas) po ukončení metody, v níž jsme jej získali. Budeme-li jej potřebovat déle, musíme si jej explicitně "přidržet" pomocí zprávy retain;
- objekt, jemuž jsme poslali zprávu retain, musíme – až pokud jej přestaneme potřebovat – uvolnit pomocí zprávy autorelease nebo release, stejně, jako objekt, jenž jsme sami vytvořili (v prvém bodě).
Připomeňme ještě v rychlosti rozdíl mezi zprávami autorelease a release: zatímco první zajistí, že objekt zanikne někdy po ukončení metody, v níž jsme jej použili, po přijetí zprávy release může objekt zaniknout ihned. Použití zprávy release je tedy obecně o něco efektivnější, ale také potenciálně nebezpečné.
Základem celého systému správy paměti v Cocoa – zapomeneme-li na chvilku na zprávu autorelease – je prachobyčejné počítání referencí: kdykoli objekt dostane zprávu retain, inkrementuje vnitřní čítač; kdykoli dostane zprávu release, opět jej dekrementuje – a pokud došel k nule, zanikne.
Skutečné kouzlo a programátorské pohodlí ale zajišťuje právě možnost používat zprávu autorelease: bez ní bychom kupříkladu nemohli rozumně (tedy bez starostí o to, kdo vlastně má objekt uvolnit) předávat objekty mezi různými metodami a podobně. Nu, a právě služby zprávy autorelease jsou ve skutečnosti zajišťovány právě třídou NSAutoreleasePool.
Základní princip je hrozně jednoduchý: instance třídy NSAutoreleasePool v podstatě není ničím jiným, nežli kontejnerem, schopným obsahovat objekty. Pošleme-li nějakému objektu zprávu autorelease, nestane se nic jiného, než že se tento objekt uloží do právě aktivního kontejneru NSAutoreleasePool; po nějakém čase standardní aplikační knihovny tento kontejner zruší – a on sám prostě při svém zániku pošle zprávu release všem objektům, jež v něm byly uloženy. Standardní aplikační knihovny pak vytvoří automaticky nový NSAutoreleasePool, a to je vlastně všechno...
Princip funkce
Ukažme si pro lepší ilustraci zdrojový kód takového velmi primitivního kontejneru, který bude – až na efektivitu a některé pomocné služby, jež si popíšeme za chvíli – fungovat prakticky stejně, jako standardní NSAutoreleasePool. Pro ukládání objektů využijeme standardní NSMutableArray a NSValue pro jejich ukládání bez "retainu"; zároveň implementujeme pomocné služby pro udržování "aktuálního poolu":
@interface DumbAutoreleasePool:NSObject { NSMutableArray *contents; DumbAutoreleasePool *previousPool; } +(void)addObject:object; // přidá objekt do aktivního poolu -(void)addObject:object; // přidá objekt do příjemce @end @implementation DumbAutoreleasePool static DumbAutoreleasePool *currentPool=nil; -init { if (!(self=[super init])) return nil; contents=[[NSMutableArray alloc] init]; // pokud byl aktuální jiný pool, zapamatujeme si jej // a sebe sama do něj uložíme (za chvíli uvidíme proč) [previousPool=currentPool addObject:self]; // a nastavíme sebe sama jako aktuální currentPool=self; return self; } +(void)addObject:object { [currentPool addObject:object]; } -(void)addObject:object { [contents addObject:[NSValue valueWithNonretainedObject:object]]; } -(void)removeObject:object { // interní služba, viz níže [contents removeObject:[NSValue valueWithNonretainedObject:object]]; } -(void)dealloc { // nejprve uvolníme všechny objekty z poolu NSEnumerator *en=[contents objectEnumerator]; id o; while (o=[en nextObject]) [[o nonretainedObjectValue] release]; // zrušíme kontejner, obnovíme current pool, a to je vše [contents release]; [currentPool=previousPool removeObject:self]; [super dealloc]; } @end
Samozřejmě, ještě bychom museli implementovat ekvivalent služby autorelease nějak takto:
@implementation NSObject (DumbAutoreleasePoolCategory) -dumbAutorelease { [DumbAutoreleasePool addObject:self]; return self; } @end
Ačkoli standardní NSAutoreleasePool je naprogramován lépe a efektivněji a nabízí některé další služby, v principu dělá přesně tohle, včetně toho hierarchického řetězení prostřednictvím proměnné previousPool, jež umožňuje pooly vzájemně vnořovat – k tomu se za chvilinku vrátíme.
Vytváření a rušení poolů
Ovšem, aby náš kód byl vůbec k něčemu dobrý, a aby služba dumbAutorelease dělala něco rozumného, musí nějaký kód nejprve vytvořit náš pool, a pak jej opět zrušit a vytvořit nový – v praxi by to, v hodně jednoduchém případě, mohlo vypadat asi nějak takhle:
int main() { int cc; // dejme tomu, že budeme číst standardní vstup while ((cc=getchar())!=EOF) { DumbAutoreleasePool *pool=[[DumbAutoreleasePool alloc] init]; ... ... tento kód může používat dumbAutorelease ... ... [pool release]; } return 0; }
V zásadě přesně tohle – ale v mnohem komplexnější podobě, kde se namísto prostého čtení standardního vstupu zpracovávají všechny možné vstupní události, od klávesnice a myši přes časovač až po zprávy, předávané mezi aplikacemi prostřednictvím distribuovaných objektů – dělá standardní kód z knihoven Cocoa s NSAutoreleasePoolem. Právě díky tomu máme kdykoli k dispozici aktuální pool, do nějž můžeme objekty ukládat pomocí zprávy autorelease, a právě díky tomu tyto objekty (pokud je, samozřejmě, někdo neretainoval) zaniknou automaticky ve chvíli, kdy končí zpracování právě aktuální události.
Hierarchie poolů
K čemu je výše zmíněné hierarchické vnořování poolů vlastně dobré? Inu, pozorní čtenáři už si to asi sami uvědomili: pokud je to zapotřebí, můžeme pooly stejně dobře vytvářet sami; zapotřebí to bude tehdy, když v rámci zpracování jediné události vytváříme příliš mnoho objektů, takže aktuální NSAutoreleasePool by příliš narostl a jeho zpracování by trvalo příliš dlouho:
... // nešikovný kód for (int i=0;i<1000000;i++) { NSString *s=[NSString stringWithFormat:@"číslo %d",i]; if (i%1) s=[s stringByAppendingString:@" je liché"]; NSArray *slova=[s componentsSeparatedByString:@" "]; ... } ...
V takovýchto případech se vyplatí použít vlastní vnořený pool, který uvolníme buďto na konci každého cyklu
... // lepší varianta for (int i=0;i<1000000;i++) { NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init]; NSString *s=[NSString stringWithFormat:@"číslo %d",i]; if (i%1) s=[s stringByAppendingString:@" je liché"]; NSArray *slova=[s componentsSeparatedByString:@" "]; ... [pool release]; // zde se uvolní vše, co se v těle cyklu // vytvořilo (a co jsme si nepřidrželi zprávou 'retain') } ...
nebo – pokud cyklus běží mnohokrát a jeho obsah je poměrně jednoduchý – až po několika průchodech:
... // ještě lepší NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init]; for (int i=0;i<1000000;i++) { ... if (i%100==99) { [pool release]; pool=[[NSAutoreleasePool alloc] init]; } } [pool release]; ...
Až si příště budeme povídat o výjimkách a jejich zpracování, pochopíme také, proč jsme vkládali vnořený pool do poolu nadřízeného: jde o to, že náš kód může "vyskočit ven" – třeba takhle, nahradíme-li dosud nepopsaný mechanismus výjimek starým dobrým goto:
... for (int i=0;i<1000000;i++) { NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init]; ... if (...) goto Out; ... [pool release]; } Out: ...
Na první pohled to vypadá velmi podezřele a nezdravě: vždyť jsme vytvořili nový pool a při výskoku jsme jej neuvolnili – nezůstanou jeho objekty "viset" v paměti "navěky" (tedy do ukončení procesu)? Inu, nezůstanou, právě díky "fintě" s uložením vnořeného poolu do poolu nadřízeného – ve chvíli, kdy jsme pool vytvořili, jsme jej uložili do poolu, jenž byl tehdy aktuální; až se tedy ten bude uvolňovat (nejspíš na konci zpracování události), zanikne i náš vnořený pool.
Obsah seriálu (více o seriálu):
- Nastal čas na kakao...
- Tak nejdřív kakao ochutnáme...
- Programovací jazyk C: velmi, velmi stručně
- Objective C: to si vysvětlíme podrobněji
- Co jsme si o Objective C ještě neřekli...
- Nastal čas na kakao - Vznik a zánik objektů
- Nastal čas na kakao - Kopírování objektů
- Nastal čas na kakao - Skryté podtřídy
- Nastal čas na kakao - Základní služby objektů
- Nastal čas na kakao - Jak správně psát v Objective C
- Nastal čas na kakao - Jak správně importovat
- Nastal čas na kakao - Podtřídy, delegáti, vkládání, jak se to rýmuje?
- Nastal čas na kakao - Využití kategorií namísto dědičnosti
- Nastal čas na kakao - Vkládání objektů a přesměrování zpráv
- Nastal čas na kakao - Inicializace a rušení objektů
- Nastal čas na kakao - Metody initWith... a designovaný inicializátor
- Nastal čas na kakao - Inicializace: tipy a triky
- Nastal čas na kakao - Accesory: přístup k proměnným instancí
- Nastal čas na kakao - Šedá je teorie, zelený je strom života...
- Nastal čas na kakao - Více o XCode: inspektory
- Nastal čas na kakao - Aplikace RSS2: datový model
- Nastal čas na kakao - Aplikace RSS: implementace datového modelu
- Nastal čas na kakao - Aplikace RSS: parsování XML
- Nastal čas na kakao - Interface Builder a uživatelské rozhraní
- Nastal čas na kakao - Interface Builder: atributy objektů
- Nastal čas na kakao - Interface Builder: atributy objektů
- Nastal čas na kakao - Druhý kontrolér a dokončení aplikace
- Nastal čas na kakao - Drobná vylepšení a zdokonalení...
- Nastal čas na kakao - Ladění
- Nastal čas na kakao - Třídy Foundation Kitu
- Nastal čas na kakao - Třídy Foundation Kitu (2)
- Nastal čas na kakao - Textové řetězce: NS(Mutable)String
- Nastal čas na kakao - Čísla, binární data a další...
- Nastal čas na kakao - Archivace objektů
- Nastal čas na kakao - Trocha magie, aneb distribuované objekty
- Nastal čas na kakao - Málem bychom zapomněli: NSAutoreleasePool
- Nastal čas na kakao - Zpracování výjimek: NSException
- Nastal čas na kakao - NSInvocation a černá magie
- Nastal čas na kakao - Kakao v Tygrovi
- Nastal čas na kakao - Notifikace: nepřímé předávání zpráv
- Nastal čas na kakao - NSUserDefaults
- Nastal čas na kakao - Co nového ve Foundation Kitu
- Nastal čas na kakao – s Intelem, s Intelem, jedeme do...
- Co nového v Xcode
- Začínáme s AppKitem
- Jak MVC v Kakau vypadá doopravdy?
- Jak MVC v Kakau vypadá doopravdy: dokončení
- Přehled tříd AppKitu
- Nastal čas na kakao - Přehled tříd AppKitu 2
- Přehled tříd AppKitu 3: zbývající třídy GUI
- Přehled tříd AppKitu 4: textový systém
- Nastal čas na kakao - Přehled tříd AppKitu 5: hlavně grafika
- Přehled tříd AppKitu 6: dokumentový systém
- Přehled tříd AppKitu 7: dokončení
- Pojmenované vlastnosti objektů
- Pojmenované vlastnosti objektů: implementace
- Pojmenované vlastnosti objektů: relace 1:N
- Pojmenované vlastnosti objektů: řazení jmen a agregační funkce
- Sledování změn objektů
- Sledování změn objektů – ukázka
- Sledování změn objektů – zdrojový kód
- Sledování změn objektů: kód modelu
- Sledování změn objektů: přímý přístup
- Kontroléry a vazby
- Vázání vazeb
- Další vazby s jednoduchým kontrolérem
- Implementace a použití převodu hodnot
- Validace hodnot
- Validace a chyby, a jedna hezká vazba...
- Práce s polem objektů
- Základní vazby NSArrayControlleru
- Převodníky, přepínače, placeholdery
- Mírná vylepšení v mezích zákona
- Objective C 2.0 - novinky z Leoparda
- NSTreeController
- Programování v Cocoa - Pár tipů a triků
- Programování v Cocoa - Základy kreslení
- Kterak nakreslit modrý obdélník...
- Další služby pro kreslení
- Obrázky a písmenka...
- Události a myš
- Lepší práce s myší
- Události klávesnice
- Input Management
- Příkazy a schránka
- Další události
- Táhni a padni
- Byli jsme na tahu; nyní padneme.
- Zvolme si, jak vhodit
- Drobnosti a chybičky
- Speciální případy tahání či házení
- Kterak táhnout něco, co neexistuje?
- Jak na sítě...
- NSURLConnection
- Safari za minutu
- Služby WebKitu
- Kakao v Leopardu
- Druhé Objective C
- Druhé Objective C: různé drobnosti
- Druhé Objective C: kategorie a protokoly
- Druhé Objective C: nový příkaz cyklu
- Druhé Objective C: atributy a accesory
- Druhé Objective C: atributy a accesory
- 64 je dvakrát 32
- Ubicumque dulce est, ibi et acidum invenies...
- Irbis: že prý žádné novinky?
- Blok sem, blok tam, nám už je to všechno jasné...
- Bloky jsou i v AppKitu
- Irbis a Foundation Kit
- Kde jsou má data?
- Kde jsou má data? V NSCache!
- Soubor, jméno, URL, jak se to rýmuje...
- Další podpora NSURL
- Zabíjení!
- A máme tady i...OS!
- Systémové prvky GUI
- Programování pro iOS 1. díl - Rozdíly mezi "i" a "Mac"
- Programování pro iOS - 2. Začínáme programovat
- Programování pro iOS - 3. základní ovladače a propojení GUI s kódem
- Programování pro iOS - 4. Varovná hlášení
- Programování pro iOS - 5. Rámce a jejich řídicí objekty
- Programování pro iOS - 6. Ukládání dat
- Programování pro iOS - 7. Správa paměti a starý restík
- Programování pro iOS - 8. Dokončení aplikace
- Programování pro iOS - 9. Jak dostat aplikaci do iPhone
- Programování pro iOS - 10. Instalace aplikace do cizího iPhone
- Programování pro iOS - 11. Jak dostat aplikaci do libovolného iPhone
- Programování pro iOS - 12. Touching!
- Programování pro iOS - 13. Kreslíme na iPhone
- Programování pro iOS - 14. Udělejme gesto
- Programování pro iOS - 15. Další gesta
- Programování pro iOS - 16. Více prstů, více zábavy
- Programování pro iOS - 17. Podpora standardních gest
- Programování pro iOS - 18. Recognizery v iOS
- Programování pro iOS - 19. Další standardní recognizery
- Programování pro iOS - 20. Co nového v iOSu
- Programování pro iOS - 21. "Multitasking"
- Programování pro iOS - 22. Nulla est honesta avaritia nisi temporis
- Programování pro iOS - 23. Jak se aktivovat, jsme-li v pozadí
- Programování pro iOS - 24. Zbývající drobnosti
- Programování pro iOS - 25. Řídicí objekty rámců
- Programování pro iOS - 26. Jak se dělá UIViewController
- Programování pro iOS - 27. Kde vzít rámce
- Programování pro iOS - 28. Základní služby
- Programování pro iOS - 29. Práce s rámci
- Programování pro iOS - 30. Rotace zařízení
- Programování pro iOS - 31. Správa paměti v rámcích
- Programování pro iOS - 32. Řídicí objekt pro tabulky
- Programování pro iOS - 33. Řídicí objekt pro strom
- Programování pro iOS - 33. Více o UINavigationControlleru
- Programování pro iOS - 35. Ještě jednou UINavigationController
- Programování pro iOS - 36. Po navigátoru taby
- Programování pro iOS - 37. Více o UITabBarControlleru
- Programování pro iOS - 38. Dokončení UITabBarControlleru
- Programování pro iOS - 39. UIPopoverController
- Programování pro iOS - 40. Další triky UIPopoverControlleru
- Programování pro iOS - 41. Zbývající služby UIPopoverControlleru
- Programování pro iOS - 42. UISplitViewController
- Programujeme v
iTunesXcode 4 - Programování pro iOS - 44. Předvolby Xcode 4
- Programování pro iOS - 45. Práce v Xcode 4
- Xcode 4: projekt a cíle
- Xcode 4: práce s cíli
- Xcode 4: Build Settings
- Xcode 4: Build Phases
- Xcode4: Build Phases podruhé
- Xcode 4: Co jsou to Build Rules?
- Xcode4: taje editoru
- Xcode4: automatické doplňování v editoru
- XIBy chyby
- Více o XIBech
- Editor XIBů
- Inspektory pro XIBy
- Vazby mezi objekty v XIBech
- Vazby mezi objekty v kódu
- Paletky Xcode pro XIBy
- Xcode 4: levý sloupec
- Xcode 4: okno Organizer
- Xcode 4: okno Organizer, část druhá
- Xcode 4: co je to Workspace?
- Xcode 4: základy schémat
- Xcode 4: akční schémata