Algorytm Borůvki

Z testwiki
Wersja z dnia 10:10, 25 mar 2024 autorstwa imported>MalarzBOT (przenoszę szablon {{Teoria grafów}} na koniec artykułu)
(różn.) ← poprzednia wersja | przejdź do aktualnej wersji (różn.) | następna wersja → (różn.)
Przejdź do nawigacji Przejdź do wyszukiwania

Szablon:Dopracować Algorytm Borůvkialgorytm wyznaczający minimalne drzewo rozpinające dla grafu nieskierowanego ważonego, o ile jest on spójny. Innymi słowy, znajduje drzewo zawierające wszystkie wierzchołki grafu, którego waga jest najmniejsza możliwa. Jest to przykład algorytmu zachłannego.

Pierwszy raz opublikowany został w 1926 roku przez Otakara Borůvkę jako metoda efektywnej konstrukcji sieci energetycznych[1][2][3]. Algorytm ten został potem ponownie wymyślony przez Choqueta w 1938 r.[4], potem przez Florka, Łukasiewicza, Perkala, Steinhausa i Zubrzyckiego w 1951 r.[5] i ostatecznie w latach 60. przez Sollina[6], od którego nazwiska często jest on nazywany „algorytmem Sollina”.

Algorytm

Przykład działania algorytmu Borůvki

Algorytm działa poprawnie przy założeniu, że wszystkie krawędzie w grafie mają różne wagi. Jeśli tak nie jest, można np. przypisać każdej krawędzi unikatowy indeks i jeśli dojdzie do porównania dwóch krawędzi o takich samych wagach, wybrać tę, która otrzymała niższy numer.

  1. Dla każdego wierzchołka v w grafie G przejrzyj zbiór incydentnych z nim krawędzi. Dołóż najlżejszą z nich do rozwiązania (zbioru E).
  2. Po tym etapie graf tymczasowego rozwiązania powinien zawierać nie więcej niż |V|2 spójnych składowych. Utwórz graf G w którym wierzchołki stanowiące spójne składowe zostaną ze sobą „sklejone” (nowe wierzchołki będziemy nazywać roboczo superwierzchołkami).
  3. Dopóki nie otrzymamy jednej spójnej składowej, wywołujemy kroki 1–2, za graf G podstawiając graf G.

Po zakończeniu algorytmu G jest minimalnym drzewem rozpinającym.

Dowód poprawności

Lemat 1

W każdym momencie działania algorytmu, oraz po jego zakończeniu w E nie będzie cyklu.

Dowód

Załóżmy nie wprost, że podczas działania algorytmu w którymś etapie pojawiła się spójna składowa, w której jest cykl. Oznaczmy ją jako S. Rozważmy następujące sytuacje:

  • S powstała przez połączenie dwóch superwierzchołków v1 i v2. Oznacza to, że do zbioru E zostały dołączone krawędzie ei i ej. Ponieważ ei została dołączona jako najlżejsza krawędź incydentna do v1 więc C(ei)<C(ej). Ale skoro ej została dołączona jako najlżejsza krawędź incydentna do v2 to musi zachodzić C(ej)<C(ei) (Pamiętajmy, że w grafie nie ma krawędzi o takiej samej wadze) – sprzeczność.
  • S powstała przez połączenie się trzech lub więcej superwierzchołków. Podzielmy powstały cykl C na następujące części: niech v1,v2,v3,,vl będą kolejnymi superwierzchołkami należącymi do C a e1,e2,,el będą kolejnymi krawędziami należącymi do C, które zostały dodane w zakończonym właśnie etapie algorytmu. W C krawędzie ei oraz superwierzchołki vi występują na przemian. Z zasady działania algorytmu możemy stwierdzić, że aby powstał taki cykl, musi zachodzić C(e1)<C(e2)<<C(el1)<C(el)<C(e1) – sprzeczność.

Lemat 2

W każdym etapie działania algorytmu otrzymujemy dla każdego superwierzchołka minimalne drzewo rozpinające

Dowód

Gdy zostanie zakończony etap 1:

Załóżmy, że istnieje taki superwierzchołek Vi, który nie jest minimalnym drzewem rozpinającym poddrzewa złożonego z wierzchołków należących do Vi. Weźmy więc takie minimalne drzewo rozpinające T. Istnieje krawędź ei taka, że eiE(Vi)ei∉E(T). Dodajmy ei. W T powstał cykl. Ponieważ ei jest incydenta do pewnego wierzchołka z tego cyklu, istnieje więc inna krawędź e'i incydentna do tego samego wierzchołka. Jednak z tego, że eiE(Vi) wynika, że C(ei)<C(e'i). Jeśli usuniemy krawędź e'i z T otrzymamy mniejsze drzewo rozpinające, co jest sprzeczne z założeniem o minimalności T.

Gdy zostanie zakończony etap 2:

Z poprawności etapu 1 wiemy, że istnieje takie wywołanie etapu 2, w którym każdy z superwierzchołków jest minimalnym drzewem rozpinającym. Jest to choćby pierwsze wywołanie. Załóżmy zatem, że dla pewnego wywołania tego etapu otrzymano superwierzchołki będące minimalnymi drzewami rozpinającymi, jednak scaliło przynajmniej dwa z nich w taki sposób, że dało się otrzymać mniejsze drzewo rozpinające.

Niech etap k-ty będzie pierwszym takim etapem, w którym coś się popsuło. Niech E'1 będzie zbiorem krawędzi przed wywołaniem etapu k, a E'2 będzie zbiorem krawędzi po jego wywołaniu. Niech T będzie minimalnym drzewem rozpinającym takim, że V(T)=V(Vi), ale że E(T)=E(Vi). Istnieje więc krawędź eiE(Vi)ei∉E(T)

Fakt: Krawędź ei została dodana podczas k-tego wywołania. (Nie może należeć do E'1, gdyż inaczej superwierzchołek do którego by należała nie byłby minimalnym drzewem rozpinającym, co jest sprzeczne z dowodem dla pierwszego etapu i założeniem, że wywołanie k-te jest najmniejszym wywołaniem, które zwróciło nieoptymalne drzewa)

Dodajmy krawędź ei do E(T). W T powstał cykl. Ponieważ ei jest najmniejszą krawędzią incydentną do pewnego superwierzchołka z tego cyklu, istnieje więc inna krawędź incydenta do tego samego superwierzchołka. Jednak jej waga jest większa niż waga krawędzi ei, zatem zastąpienie jej krawędzią ei da nam mniejsze drzewo rozpinające co jest sprzeczne z założeniem o optymalności T.

Złożoność obliczeniowa

Można udowodnić, że każdym kolejnym wywołaniu liczba wierzchołków zmaleje co najmniej dwukrotnie – zatem wywołań tych będzie co najwyżej logV. Złożoność obliczeniowa całości zależy więc od sposobu implementacji kroków 1, 2 algorytmu. Zastosowanie w implementacji struktury zbiorów rozłącznych pozwala osiągnąć złożoność na poziomie O(ElogV). Można zmodyfikować go tak, aby osiągnąć złożoność O(ElogVV) – algorytm działa wtedy znacznie szybciej dla grafów rzadkich; dla grafów gęstych modyfikacja nie daje dużych zysków czasowych.


Zobacz też

Przypisy

Szablon:Przypisy

Linki zewnętrzne

Szablon:Teoria grafów