Povodňová výplň - Flood fill
tento článek potřebuje další citace pro ověření.Srpna 2009) (Zjistěte, jak a kdy odstranit tuto zprávu šablony) ( |
Povodňová výplň, také zvaný výplň semen, je algoritmus která určuje oblast připojeno do daného uzlu ve vícerozměrném pole. Používá se v nástroji pro výplň „kbelík“ malířské programy k vyplnění propojených, podobně zbarvených oblastí jinou barvou a ve hrách, jako je Jít a Minolovka pro určení, které kusy jsou vyklizeny.
Algoritmus
Algoritmus povodňových výplní má tři parametry: počáteční uzel, cílovou barvu a náhradní barvu. Algoritmus vyhledá všechny uzly v poli, které jsou připojeny k počátečnímu uzlu cestou cílové barvy, a změní je na náhradní barvu. Existuje mnoho způsobů, jak lze strukturovat algoritmus povodňové výplně, ale všechny používají a fronta nebo zásobník datová struktura, výslovně nebo implicitně.
V závislosti na tom, zda považujeme uzly dotýkající se v rozích za spojené nebo ne, máme dvě varianty: osmicestné a čtyřcestné.
Zásobníková rekurzivní implementace (čtyřcestná)
Jeden implicitně založený na zásobníku (rekurzivní ) implementace fill-fill (pro dvourozměrné pole) probíhá následovně:
Povodeň (uzel, cílová barva, náhradní barva): 1. Pokud cílová barva je rovný náhradní barva, vrátit se. 2. Jestli je barva uzel se nerovná cílová barva, vrátit se. 3. Jinak Nastavte barvu uzel na náhradní barva. 4. Proveďte Povodeň (jeden krok na jih od uzel, cílová barva, náhradní barva). Provést Povodeň (jeden krok na sever od uzel, cílová barva, náhradní barva). Provést Povodeň (jeden krok na západ od uzel, cílová barva, náhradní barva). Provést Povodeň (jeden krok na východ od uzel, cílová barva, náhradní barva). 5. Vraťte se.
Ačkoli je snadno pochopitelné, implementace výše použitého algoritmu je nepraktická v jazycích a prostředích, kde je prostor zásobníku vážně omezen (např. Java applety ).
Alternativní implementace
Explicitně implementace založená na frontě (někdy nazývaná „Algoritmus lesního požáru“)[1]) je uveden níže v pseudokódu. Je to podobné jednoduchému rekurzivnímu řešení, až na to, že místo rekurzivního volání tlačí uzly na a fronta pro spotřebu:
Povodeň (uzel, cílová barva, náhradní barva): 1. Pokud cílová barva je rovný náhradní barva, vrátit se. 2. Pokud je barva uzel se nerovná cílová barva, vrátit se. 3. Nastavte barvu uzel na náhradní barva. 4. Nastavit Q do prázdné fronty. 5. Přidat uzel do konce roku Q. 6. Zatímco Q není prázdný: 7. Nastav n se rovná prvnímu prvku Q. 8. Odstraňte první prvek z Q. 9. Pokud je barva uzlu na západ od n cílová barva, nastavte barvu tohoto uzlu na náhradní barvu a přidejte tento uzel na konec Q. 10. Pokud je barva uzlu na východ od n je cílová barva, nastavte barvu tohoto uzlu na náhradní barvu a přidejte tento uzel na konec Q. 11. Pokud je barva uzlu na sever od n cílová barva, nastavte barvu tohoto uzlu na náhradní barvu a přidejte tento uzel na konec Q. 12. Pokud je barva uzlu na jih od n cílová barva, nastavte barvu tohoto uzlu na náhradní barvu a přidejte tento uzel na konec Otázka 13. Pokračujte ve smyčce až do Q je vyčerpaný. 14. Návrat.
Praktické implementace určené k vyplňování obdélníkových oblastí mohou jako optimalizaci použít smyčku pro západní a východní směr, aby se zabránilo režii správy zásobníku nebo fronty:
Povodeň (uzel, cílová barva, náhradní barva): 1. Pokud cílová barva je rovný náhradní barva, vrátit se. 2. Pokud je barva uzel se nerovná cílová barva, vrátit se. 3. Nastavit Q do prázdné fronty. 4. Přidat uzel na Q. 5. Pro každý prvek N z Q: 6. Nastavit w a E rovná N. 7. Pohyb w na západ až do barvy uzlu na západ od w již neodpovídá cílová barva. 8. Pohyb E na východ až do barvy uzlu na východ od E již neodpovídá cílová barva. 9. Pro každý uzel n mezi w a E: 10. Nastavit barvu n na náhradní barva.11. Pokud je barva uzlu na sever od n je cílová barva, přidejte tento uzel do Q.12. Pokud je barva uzlu na jih od n je cílová barva, přidejte tento uzel do Q.13. Pokračujte ve smyčce, dokud Q je vyčerpána. Vrátit se.
Přizpůsobení algoritmu k použití dalšího pole k uložení tvaru oblasti umožňuje zobecnění pokrýt „fuzzy“ zaplavení, kde se prvek může lišit až o stanovenou prahovou hodnotu od zdrojového symbolu. Použití tohoto dalšího pole jako alfa kanál umožňuje okrajům vyplněné oblasti trochu hladce splynout s nevyplněnou oblastí.[Citace je zapotřebí ]
Metoda pevné paměti (metoda výplně vpravo)
Existuje metoda, která pro v podstatě nepoužívá žádnou paměť čtyři připojené regiony tím, že předstírají, že jsou malíři, kteří se snaží namalovat region, aniž by se namalovali do rohu. Toto je také metoda řešení bludišť. Prozkoumají se čtyři pixely tvořící primární hranici, aby se zjistilo, jaké kroky je třeba provést. Malíř se mohl ocitnout v jedné z několika podmínek:
- Všechny čtyři hraniční pixely jsou vyplněny.
- Tři z hraničních pixelů jsou vyplněny.
- Dva z hraničních pixelů jsou vyplněny.
- Jeden hraniční pixel je vyplněn.
- Nulové hraniční pixely jsou vyplněny.
Tam, kde je třeba sledovat cestu nebo hranici, se použije pravidlo pravé ruky. Malíř sleduje region tím, že položí svou pravou ruku na zeď (hranice regionu) a postupuje kolem okraje regionu, aniž by sejmul ruku.
V případě č. 1 malíř namaluje (vyplní) pixel, na kterém malíř stojí, a zastaví algoritmus.
V případě č. 2 existuje cesta vedoucí z oblasti. Namalujte pixel, na kterém malíř stojí, a pohybujte se ve směru otevřené cesty.
V případě č. 3 dva hraniční pixely definují cestu, která, pokud bychom namalovali aktuální pixel, nám může bránit v návratu na druhou stranu cesty. Potřebujeme „značku“, abychom definovali, kde jsme a kterým směrem míříme, abychom zjistili, zda se někdy dostaneme zpět k přesně stejnému pixelu. Pokud jsme již vytvořili takovou „značku“, zachováme si naši předchozí značku a přejdeme k dalšímu pixelu podle pravidla pravé ruky.
Značka se používá pro první hranici dvou pixelů, která se vyskytne, aby si pamatovala, kde průchod začal a jakým směrem se malíř pohyboval. Pokud se se značkou znovu setkáte a malíř cestuje stejným směrem, pak malíř ví, že je bezpečné namalovat čtverec značkou a pokračovat stejným směrem. Je to proto, že (přes nějakou neznámou cestu) lze v budoucnu dosáhnout a vymalovat pixely na druhé straně značky. Značka je odstraněna pro budoucí použití.
Pokud malíř narazí na značku, ale jde jiným směrem, došlo k nějaké smyčce, která způsobila, že se malíř vrátil ke značce. Tato smyčka musí být odstraněna. Značka je zvednuta a malíř poté postupuje ve směru, který je známkou označenou dříve, pomocí pravítka pro hranici hranice (podobné pravidlu pro pravou ruku, ale pomocí malířovy levé ruky). To pokračuje, dokud není nalezen průnik (se třemi nebo více otevřenými hraničními pixely). Malíř, který stále používá pravidlo levé ruky, nyní hledá jednoduchý průchod (vytvořený dvěma hraničními pixely). Po nalezení této cesty hranice dvou pixelů je tento pixel vymalován. To přeruší smyčku a umožní algoritmu pokračovat.
V případě č. 4 musíme zkontrolovat protilehlé 8 spojené rohy, abychom zjistili, zda jsou vyplněné nebo ne. Pokud je vyplněna jedna nebo obě, vytvoří se tím křižovatka mnoha cest a nelze ji vyplnit. Pokud jsou obě prázdné, pak lze aktuální pixel vymalovat a malíř se může pohybovat podle pravidla pravé ruky.
Algoritmus vyměňuje čas za paměť. U jednoduchých tvarů je to velmi efektivní. Pokud je však tvar složitý s mnoha funkcemi, algoritmus stráví velké množství času sledováním okrajů oblasti a snaží se zajistit, aby bylo možné všechny vymalovat.
Tento algoritmus byl poprvé komerčně dostupný v roce 1981 na systému zpracování obrazu Vicom vyráběném společností Vicom Systems, Inc. Klasický algoritmus rekurzivní zaplavení byl k dispozici také na tomto systému.
Pseudo kód
Toto je implementace pseudokódu optimálního algoritmu zaplnění paměti s pevnou pamětí napsaného ve strukturované angličtině:
Proměnné:cur, mark a mark2 každý obsahuje buď souřadnice pixelu, nebo nulovou hodnotu
POZNÁMKA: pokud je značka nastavena na null, nevymažte její předchozí hodnotu souřadnic. Mějte tyto souřadnice k dispozici, abyste si je v případě potřeby mohli vyvolat.
cur-dir, mark-dir a mark2-dir každý drží směr (doleva, doprava, nahoru nebo dolů) backtrack a findloop každý hold booleovské hodnoty počet je celé číslo
Algoritmus:
(POZNÁMKA: Všechny směry (přední, zadní, levý, pravý) jsou relativní k cur-dir)
nastavit cur na počáteční pixely set cur-dir na výchozí směr clear mark a mark2 (nastavit hodnoty na null) nastavit backtrack a findloop na falsezatímco přední pixel je prázdný dělat posunout se vpředskončit chvíliskočit na STARTMAIN LOOP: posunout se vpřed -li pravý pixel je prázdný pak -li backtrack je pravda a findloop je false a buď přední pixel nebo levý pixel je prázdný pak nastavit findloop na true skončit, pokud odbočit vpravo PAINT: posunout se vpřed skončit, pokudSTART: soubor počet na počet ne diagonálně přilehlých pixelů vyplněných (POUZE přední / zadní / levý / pravý) -li počet není 4 pak dělat odbočit vpravo zatímco přední pixel je prázdný dělat odbočit vlevo zatímco přední pixel je vyplněn skončit, pokud přepínač počet případ 1 -li backtrack je pravda pak nastavit findloop na true jinak pokud findloop je pravda pak -li známka je null pak obnovit značku skončit, pokud jinak pokud přední levý pixel a zadní levý pixel jsou prázdné pak zrušte označení výplně kurzorovým skokem na PAINT skončit, pokud koncový případ 2 -li zadní pixel je vyplněn pak -li přední levý pixel není vyplněn pak zrušte označení výplně kurzorovým skokem na PAINT skončit, pokud jinak pokud značka není nastavena pak set mark to cur set mark-dir to cur-dir clear mark2 set findloop and backtrack to false jiný -li značka 2 není nastavena pak -li Cur je u značky pak -li cur-dir je stejný jako mark-dir pak vymazat značku otočit výplň kurzí skok na PAINT jiný nastavit backtrack na true set findloop na false nastavit cur-dir na mark-dir skončit, pokud jinak pokud findloop je pravda pak nastavit mark2 na cur nastavit mark2-dir na cur-dir skončit, pokud jiný -li Cur je u značky pak nastavit cur na mark2 nastavit cur-dir na mark2-dir vymazat značku a mark2 nastavit backtrack na falešný obrat kolem fill cur skočit na PAINT jiný pokud cur na mark2 pak nastavit značku na cur nastavit cur-dir a mark-dir na mark2-dir clear mark2 skončit, pokud skončit, pokud skončit, pokud koncový případ 3 jasná značka vyplňte cur skok na PAINT koncový případ 4 vyplňte cur hotový koncový případ koncový spínačkonec HLAVNÍ SLUČKA
Scanline výplň
Algoritmus lze zrychlit vyplněním řádků. Místo toho, aby posunul každou potenciální budoucí souřadnici pixelů na zásobníku, zkontroluje sousední čáry (předchozí a další), aby našel sousední segmenty, které mohou být vyplněny v budoucím průchodu; souřadnice (začátek nebo konec) úsečky jsou vloženy do zásobníku. Ve většině případů je tento algoritmus skenovací řádky alespoň o řád rychlejší než ten na pixel.
Účinnost: každý pixel je zkontrolován jednou.
Vektorové implementace
Verze 0.46 z Inkscape zahrnuje nástroj pro výplň kbelíku, který poskytuje výstup podobný běžným bitmapovým operacím a skutečně používá jednu: plátno je vykresleno, na vybrané ploše je provedena povodňová výplň a výsledek je pak vysledován zpět na cestu. Využívá koncept a okrajová podmínka.
Chování ve velkém měřítku
Primární technika používaná k řízení zaplavení bude buď datově nebo procesně. Přístup zaměřený na data může ke sledování počátečních pixelů, které je třeba zkontrolovat, použít buď zásobník, nebo frontu. Algoritmus zaměřený na proces musí nutně používat zásobník.
Algoritmus čtyřcestného zaplavení, který používá techniku sousednosti a frontu, protože její úložiště počátečních pixelů přináší výplň ve tvaru kosočtverce.
Účinnost: 4 pixely zkontrolovány pro každý naplněný pixel (8 pro 8cestnou výplň).
Algoritmus čtyřcestného zaplavení, který používá techniku sousednosti a zásobník jako úložiště počátečních pixelů, poskytuje lineární výplň s chováním „mezery vyplněné později“. Tento přístup lze vidět zejména ve starších 8bitových počítačových hrách, jako jsou hry vytvořené pomocí Tvůrce grafického dobrodružství.
Účinnost: 4 pixely zkontrolovány pro každý naplněný pixel (8 pro 8cestnou výplň).
Viz také
- Šířka první vyhledávání
- Hledání do hloubky
- Traverz grafu
- Označování připojených komponent
- Dijkstrův algoritmus
externí odkazy
- Didaktická implementace skriptové polygonové výplně Javascript od Guilherme Polo.
- Ukázkové implementace pro rekurzivní a nerekurzivní, klasickou a skenovanou výplň autor: Lode Vandevenne.
- Implementace Flash Flood fill, Emanuele Feronato.
- QuickFill: Efektivní algoritmus zaplavení., John R. Shaw.
- FloodSpill: open-source algoritmus plnění povodní pro C # autor: Paweł Ślusarczyk
Reference
- ^ Torbert, Shane (2016). Aplikovaná informatika (2. vyd.). Springer (zveřejněno 01.06.2016). p. 158. doi:10.1007/978-3-319-30866-1. ISBN 978-3-319-30864-7. LCCN 2016936660. Archivováno od původního dne 2016-12-21.