Jak rozpoznawać algorytmy o kwadratowej złożoności czasowej?

Wyobraźmy sobie, że analizujemy pewien algorytm o kwadratowej złożoności czasowej. Niech funkcja wyznacza dokładną liczbę operacji wykonywanych przez program w zależności od ilości danych wejściowych. Przyjmijmy, że ma ona następującą postać:

,.

przy czym jest pewną stałą. Jak w takim razie może być zbudowany algorytm? Spróbujmy zapisać funkcję w nieco inny sposób, a mianowicie:

Przedstawimy kilka propozycji programów, w których zastosowanie znajdą algorytmy o kwadratowej złożoności czasowej.

Obliczanie sumy długości boków i przekątnych figury

Pierwszy program przyjmuje na wejście współrzędne geometryczne punktów, , , , będących wierzchołkami figury wypukłejfigura wypukłafigury wypukłej. Na podstawie tych danych wyznaczane są długości wszystkich odcinków , przy czym oraz . Otrzymane wartości są następnie sumowane, dzięki czemu otrzymujemy pożądaną sumę długości boków i przekątnych figury.

R1GixojeLFAsQ
Źródło: Contentplus.pl Sp. z o.o., licencja: CC BY-SA 3.0.

Zapiszmy odpowiedni algorytm w postaci pseudokodu. Niech Punkty oznaczają zbiór n punktów podanych jako dane wejściowe. Dodatkowo niech instrukcja Punkty[i] oznacza odwołanie się do współrzędnych i-tego punktu.

Linia 1. suma ← 0. Linia 2. i ← 1. Linia 4. Dopóki i otwórz nawias ostrokątny znak równości otwórz nawias okrągły n minus 1 zamknij nawias okrągły wykonuj. Linia 5. j ← i plus 1. Linia 7. Dopóki j otwórz nawias ostrokątny znak równości n wykonuj. Linia 8. suma ← suma plus odleglosc otwórz nawias okrągły Punkty otwórz nawias kwadratowy i zamknij nawias kwadratowy przecinek Punkty otwórz nawias kwadratowy j zamknij nawias kwadratowy zamknij nawias okrągły. Linia 9. j ← j plus 1. Linia 11. i ← i plus 1. Linia 13. wypisz otwórz nawias okrągły suma zamknij nawias okrągły.

Dla każdego punktu spośród n podanych na wejściu obliczana jest odległość między nim a pozostałymi punktami. Dodatkowo zadbaliśmy o to, aby odległość między każdą parą różnych punktów była liczona dokładnie raz. Dzięki temu w trakcie rozpatrywania i-tego z kolei punktu, obliczamy odległość dokładnie (n - i) razy. W rezultacie funkcja opisująca liczbę wykonywanych operacji przybierze następującą postać:

Po skorzystaniu ze wzoru na sumę ciągu arytmetycznego otrzymujemy:

Udowodniliśmy zatem, że nasz algorytm ma kwadratową złożoność czasową.

Rysowanie figur w oknie konsoli

Kolejnym pomysłem na zastosowanie algorytmu o czasowej złożoności kwadratowej jest wypisywanie w oknie konsoli kwadratu zbudowanego ze znaków #. Jego bok składa się z  znaków, przy czym o wartości decyduje użytkownik na początku działania programu.

RFaS6IpLKTIWI
Źródło: Contentplus.pl Sp. z o.o., licencja: CC BY-SA 3.0.

Zapiszmy zatem algorytm w postaci pseudokodu.

Linia 1. wczytaj otwórz nawias okrągły n zamknij nawias okrągły. Linia 2. i ← 1. Linia 4. Dopóki i otwórz nawias ostrokątny znak równości n wykonuj. Linia 5. j ← 1. Linia 7. Dopóki j otwórz nawias ostrokątny znak równości n wykonuj. Linia 8. wypisz otwórz nawias okrągły cudzysłów kratka cudzysłów zamknij nawias okrągły. Linia 9. j ← j plus 1. Linia 11. wypisz otwórz nawias okrągły cudzysłów lewy ukośnik n cudzysłów zamknij nawias okrągły. Linia 12. i ← i plus 1.

Dosyć nietypowa może się wydać instrukcja wypisz("\n"). Symbol \n jest jednym z symboli sterujących. Mówi on funkcji wypisującej znaki, że należy przejść do nowej linii. Rezultat jest taki sam jak po naciśnięciu klawisza [Enter] w czasie pracy z edytorem tekstu.

Udowodnienie, że zastosowany algorytm ma kwadratową złożoność czasową jest proste. Zauważmy, że skoro bok kwadratu składa się z  znaków #, to musimy ich wypisać . W konsekwencji należy wykonać około operacji; oznacza to, że mamy do czynienia z kwadratową złożonością czasową.

Sortowanie elementów

R1K8S5r88MxPd1
Źródło: Contentplus.pl Sp. z o.o., licencja: CC BY-SA 3.0.

Ostatnim przykładem jest sortowanie liczb podanych na wejściu przez użytkownika. Ponieważ w czasie obecnej lekcji nie omawiamy sposobów przechowywania danych, nie będziemy zajmować się tym zagadnieniem. Ograniczymy się do przedstawienia idei algorytmu sortowania.

Spośród liczb będzie wybierana najmniejsza. Zostanie ona usunięta z przeszukiwanego zbioru i wypisana na wyjściu. Następnie te same operacje będą powtarzane dla zbiorów złożonych z  elementów.

R18gP1EFgkdHx
Źródło: Contentplus.pl Sp. z o.o., licencja: CC BY-SA 3.0.

Niżej znajduje się algorytm przedstawiony w postaci pseudokodu. Niech Liczby oznaczają zbiór n liczb podanych na wejściu. Natomiast instrukcja Liczby[i] oznacza odwołanie się do i-tej liczby.

Linia 1. Dopóki n zamknij nawias ostrokątny 0 wykonuj. Linia 2. minimum ← Punkty otwórz nawias kwadratowy 1 zamknij nawias kwadratowy. Linia 3. i ← 2. Linia 5. Dopóki i otwórz nawias ostrokątny znak równości n wykonuj. Linia 6. Jeżeli minimum zamknij nawias ostrokątny Punkty otwórz nawias kwadratowy i zamknij nawias kwadratowy. Linia 7. minimum ← Punkty otwórz nawias kwadratowy i zamknij nawias kwadratowy. Linia 9. i ← i plus 1. Linia 11. wypisz otwórz nawias okrągły minimum zamknij nawias okrągły. Linia 12. usun podkreślnik ze podkreślnik zbioru otwórz nawias okrągły minimum zamknij nawias okrągły. Linia 13. n ← n minus 1.

Ile operacji wykona program? Znalezienie najmniejszej wartości w zbiorze -elementowym wymaga wykonania porównań. Stosując omawiany algorytm, będziemy wyszukiwać najmniejszy element kolejno w zbiorze -elementowym, następnie w -elementowym i tak dalej. W rezultacie funkcja opisująca liczbę wykonywanych operacji przybierze następującą postać:

.

Jest to suma ciągu arytmetycznego o wyrazie początkowym równym i różnicy wynoszącej . Korzystając z wzoru na sumę takiego ciągu, otrzymujemy:

.

Udowodniliśmy zatem, że złożoność czasowa przedstawionego algorytmu sortowania jest kwadratowa.

Przybliżony czas realizacji algorytmów o złożoności kwadratowej

Poznaliśmy i nauczyliśmy się rozpoznawać już trzy złożoności czasowe: stałą, liniową i kwadratową. Korzystając z przybliżenia, zgodnie z którym współczesnym komputerom wykonanie miliarda () operacji zajmuje około sekundę, porównajmy czas działania programów bazujących na algorytmach o różnych złożonościach czasowych.

Liczba danych wejściowych

Czas realizacji algorytmu

Czas realizacji algorytmu

sekundy

sekundy

sekundy

sekundy

sekundy

sekund minut

sekundy

sekund dni

sekund

sekund lat

sekund minut

sekund mln lat

Dla bardzo małej liczby danych wejściowych różnica między algorytmami liniowymi a kwadratowymi jest naprawdę niewielka. Jednak wraz ze wzrostem liczby danych czas działania programu, w którym zaimplementowano algorytm o kwadratowej złożoności czasowej, rośnie w bardzo dużym tempie. Właśnie dlatego tak ważna jest optymalizacja czasowa programów.

Ciekawostka

Kryptologia, czyli dziedzina wiedzy zajmująca się przekazywaniem informacji w sposób zabezpieczony przed osobami niepowołanymi, bardzo mocno opiera się na powyższych zależnościach czasowych. Aby złamać 12‑znakowe hasło złożone wyłącznie z małych liter alfabetu angielskiego, trzeba poświęcić aż 30 lat.

Słownik

figura wypukła
figura wypukła

figura geometryczna, w której zawarty jest każdy odcinek łączący jej dwa dowolne punkty; przykładowymi figurami wypukłymi są trójkąt, elipsa i prostokąt