Grahamovo skenování - Graham scan
Grahamovo skenování je metoda hledání konvexní obal konečné sady bodů v rovině s časová složitost Ó (n log n). Je pojmenován po Ronald Graham, který publikoval původní algoritmus v roce 1972.[1] Algoritmus najde všechny vrcholy konvexního trupu seřazené podél jeho hranice. Používá a zásobník efektivně detekovat a odstranit konkávnosti na hranici.
Algoritmus
Prvním krokem v tomto algoritmu je nalezení bodu s nejnižší souřadnicí y. Pokud nejnižší souřadnice y existuje ve více než jednom bodě sady, měl by být vybrán bod s nejnižší souřadnicí x z kandidátů. Nazvěte tento bod P. Tento krok trvá Ó (n), kde n je počet dotyčných bodů.
Dále je nutné setřídit sadu bodů ve vzestupném pořadí podle úhlu, ve kterém se nacházejí, a bodu P udělejte s osou x. Jakékoli univerzální třídicí algoritmus je k tomu například vhodný heapsort (což je O (n log n)).
Řazení podle úhlu nevyžaduje výpočet úhlu. Je možné použít jakoukoli funkci úhlu, která je v interval . Kosinus lze snadno vypočítat pomocí Tečkovaný produkt, nebo lze použít sklon čáry. Pokud jde o číselnou přesnost, může funkce porovnání použitá algoritmem třídění použít znaménko křížový produkt určit relativní úhly.
Algoritmus pokračuje zvážením každého z bodů v seřazeném poli v pořadí. U každého bodu se nejprve určí, zda cestování ze dvou bodů bezprostředně předcházejících tomuto bodu představuje odbočku doleva nebo doprava. Pokud odbočíte doprava, předposlední bod není součástí konvexního trupu a leží „uvnitř“. Stejné určení se pak provede pro množinu posledního bodu a dva body, které bezprostředně předcházejí bodu, o kterém se zjistilo, že byl uvnitř trupu, a opakuje se, dokud nenarazí na množinu „otočení doleva“, kdy se algoritmus posune dál do dalšího bodu v sadě bodů v seřazeném poli minus všechny body, u kterých bylo zjištěno, že jsou uvnitř trupu; není třeba tyto body znovu zvažovat. (Pokud jsou v kterékoli fázi tři body kolineární, lze se rozhodnout buď zahodit, nebo to nahlásit, protože v některých aplikacích je nutné najít všechny body na hranici konvexního trupu.)
Opět platí, že určení, zda tři body tvoří „odbočku doleva“ nebo „odbočku doprava“, nevyžaduje výpočet skutečného úhlu mezi dvěma úsečkami a lze ji ve skutečnosti dosáhnout pouze jednoduchou aritmetikou. Za tři body , a , spočítat z-koordinátor z křížový produkt ze dvou vektory a , což je dáno výrazem . Pokud je výsledek 0, body jsou kolineární; pokud je kladná, tvoří tři body orientaci „doleva“ nebo proti směru hodinových ručiček, jinak „doprava“ nebo orientaci ve směru hodinových ručiček (pro body očíslované proti směru hodinových ručiček).
Tento proces se nakonec vrátí do bodu, ve kterém začal, kdy je algoritmus dokončen a zásobník nyní obsahuje body na konvexním trupu v pořadí proti směru hodinových ručiček.
Časová složitost
Třídění bodů má časovou složitost O (n log n). I když se může zdát, že časová složitost smyčky je O (n2), protože pro každý bod se vrací zpět, aby zkontroloval, zda některý z předchozích bodů „zatočil doprava“, je to vlastně O (n), protože každý bod je v určitém smyslu považován maximálně dvakrát. Každý bod se může objevit jako bod pouze jednou v "levé zatáčce" (protože algoritmus postupuje do dalšího bodu poté) a jako bod v "zatáčce doprava" (protože bod je odebrán). Celková časová složitost je tedy O (n log n), protože čas na třídění dominuje času na skutečný výpočet konvexního trupu.
Pseudo kód
Níže uvedený kód používá funkci ccw: ccw> 0, pokud se tři body otočí proti směru hodinových ručiček, ve směru hodinových ručiček, pokud ccw <0, a kolineární, pokud ccw = 0. (V reálných aplikacích, pokud jsou souřadnice libovolná reálná čísla, funkce vyžaduje přesné srovnání čísel s plovoucí desetinnou čárkou a je třeba si dávat pozor na numerické singularity pro „téměř“ kolineární body.)
Poté nechte výsledek uložit do zásobník
.
nechat bodů být seznam bodůnechat stack = empty_stack ()nalézt nejnižší souřadnice y a nejvíce vlevo, nazývaná P0třídit body podle polárního úhlu s P0, pokud má několik bodů stejný polární úhel, pak se držte jen nejvzdálenějšípro směřovat v points: # vyklopíme poslední bod ze zásobníku, pokud dosáhneme tohoto bodu otočením ve směru hodinových ručiček zatímco počet stoh> 1 a ccw(next_to_top (stack), top (stack), point) <= 0: pop zásobník tlačit směřovat na zásobníkkonec
Zásobník nyní obsahuje konvexní trup, kde jsou body orientovány proti směru hodinových ručiček a P0 je první bod.
Tady, next_to_top ()
je funkce pro vrácení položky o jednu položku pod horní část stohu, aniž by došlo ke změně stohu, a podobně, horní()
pro vrácení nejvyššího prvku.
Tento pseudokód je převzat z Úvod do algoritmů.
Poznámky
Stejná základní myšlenka funguje také v případě, že je vstup tříděn na souřadnici x namísto úhlu a trup je vypočítán ve dvou krocích, přičemž se vytvoří horní a dolní část trupu. Tuto úpravu vymyslel A. M. Andrew[2] a je znám jako Andrewův monotónní řetězový algoritmus. Má stejné základní vlastnosti jako Grahamovo skenování.[3]
Technika zásobníku použitá při Grahamově skenování je velmi podobná technice pro všechny nejbližší menší hodnoty problém a paralelní algoritmy pro všechny nejbližší menší hodnoty mohou být také použity (jako Grahamův sken) k efektivnímu výpočtu konvexních trupů seřazených posloupností bodů.[4]
Numerická robustnost
Numerická robustnost je problém řešit v algoritmech, které používají konečnou přesnost plovoucí bod počítačová aritmetika. Dokument z roku 2004 analyzoval jednoduchou přírůstkovou strategii, kterou lze použít zejména pro implementaci Grahamova skenu.[5] Uvedeným cílem příspěvku nebylo konkrétně analyzovat algoritmus, ale spíše poskytnout učebnicový příklad toho, co a jak může selhat kvůli výpočtům s plovoucí desetinnou čárkou v výpočetní geometrie.[5] Později D. Jiang a N. F. Stewart[6] k tomu rozpracován a pomocí zpětná analýza chyb učinil dva hlavní závěry. První je, že konvexní trup je a dobře podmíněný problém, a proto lze očekávat algoritmy, které vytvoří odpověď v rozumné míře chyby. Za druhé, prokazují, že modifikace Grahamova skenování, které nazývají Graham-Fortune (zahrnující myšlenky na Steven Fortune pro numerickou stabilitu[7]) překonává problémy s konečnou přesností a nepřesnými údaji „v jakémkoli rozsahu je to možné“.
Viz také
Reference
- ^ Graham, R.L. (1972). „Efektivní algoritmus pro určení konvexního trupu konečné planární množiny“ (PDF). Dopisy o zpracování informací. 1 (4): 132–133. doi:10.1016/0020-0190(72)90045-2.
- ^ Andrew, A. M. (1979). "Další efektivní algoritmus pro konvexní trupy ve dvou rozměrech". Dopisy o zpracování informací. 9 (5): 216–219. doi:10.1016/0020-0190(79)90072-3.
- ^ De Berg, Mark; Cheong, Otfried; Van Kreveld, Marc; Overmars, Mark (2008). Algoritmy a aplikace výpočetní geometrie. Berlín: Springer. str.2 –14. doi:10.1007/978-3-540-77974-2. ISBN 978-3-540-77973-5.
- ^ Berkman, Omer; Schieber, Baruch; Vishkin, Uzi (1993). Msgstr "Optimální dvojité logaritmické paralelní algoritmy založené na nalezení všech nejbližších menších hodnot". Journal of Algorithms. 14 (3): 344–370. CiteSeerX 10.1.1.55.5669. doi:10.1006 / jagm.1993.1018..
- ^ A b Kettner, Lutz; Mehlhorn, Kurt; Pion, Sylvain; Schirra, Stefan; Yap, Chee (2008). „Učebny - příklady problémů s robustností v geometrických výpočtech“ (PDF). Výpočetní geometrie. 40 (1): 61–78. doi:10.1016 / j.comgeo.2007.06.003. (Starší verze byla nahlášena v roce 2004 na ESA'2004)
- ^ D. Jiang a N. F. Stewart, Zpětná analýza chyb ve výpočetní geometrii Archivováno 09.08.2017 na Wayback Machine, Výpočetní věda a její aplikace - ICCSA 2006 Svazek 3980 série Přednášky z informatiky, str. 50-59
- ^ Fortune, S. Stabilní údržba triangulací množiny bodů ve dvou rozměrech. Pokračování 30. ročníku IEEE Symposium on Foundations of Computer ScienceVol. 30, 494-499, 1989.
Další čtení
- Cormen, Thomas H.; Leiserson, Charles E.; Rivest, Ronald L.; Stein, Clifford (2001) [1990]. „33.3: Nalezení konvexního trupu“. Úvod do algoritmů (2. vyd.). MIT Press a McGraw-Hill. 949–955. ISBN 0-262-03293-7.