Jednoczesne wyszukiwanie minimum i maksimum¶
Zdarza się i tak, że potrzebujemy znaleźć wartość minimalną i maksymalną jednocześnie, najlepiej za jednym razem. Możemy oczywiście osobno wyszukać minimum i maksimum korzystając ze standardowego algorytmu. Być może jednak da się to zrobić lepiej, wydajniej? Na to pytanie postaramy się odpowiedzieć. Zacznijmy od formalnej specyfikacji.
Specification¶
Input¶
- \(n\) — liczba naturalna, liczba elementów w tablicy
- \(A[1..n]\) — tablica \(n\) wartości całkowitych
Output¶
- Największa oraz najmniejsza wartość z tablicy \(A\)
Example¶
Input¶
Output¶
Solution naiwne¶
Zacznijmy od rozwiązania naiwnego. Pomysł jest następujący: zastosujmy standardowy algorytm do znajdowania minimum i maksimum. Na początku jako tymczasowe minimum i maksimum przyjmiemy wartość pierwszego elementu z tablicy. Następnie przejdziemy pętlą przez kolejne elementy. Każdy element będziemy porównywać z dotychczasowymi wartościami minimum i maksimum, dokonując odpowiednich zamian w razie potrzeby.
Pseudocode¶
funckja SzukajMinMax(n, A):
1. min := A[1]
2. max := A[1]
3. Od i := 2 do n, wykonuj:
4. Jeżeli min > A[i], to:
5. min := A[i]
6. Jeżeli max < A[i], to:
7. max := A[i]
8. Zwróć min, max
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["SzukajMinMax(n, A)"]) --> K1["min := A[1]
max := A[1]
i := 1"]
K1 --> K3{i <= n}
K3 -- TRUE --> K4{"min > A[i]"}
K4 -- TRUE --> K5["min := A[i]"]
K5 --> K6{"max < A[i]"}
K4 -- FALSE --> K6
K6 -- TRUE --> K7["max := A[i]"]
K7 --> K3i[i := i + 1]
K6 -- FALSE --> K3i
K3i --> K3
K3 -- FALSE ---> K8[/Zwróć min, max/]
K8 ----> STOP([STOP])
Complexity¶
\(O(2n)\)
Mamy jedną pętlę, ale dwa porównania wewnątrz niej. W takim razie dla każdego przebiegu pętli wykonujemy dwa porównania, łącznie wykonujemy ich więc w przybliżeniu \(2n\), co w praktyce daje nam złożoność liniową.
Solution optymalne¶
Podejdźmy do problemu od innej strony. Zastanówmy się, jak możemy przygotować sobie dane, aby ułatwić sobie pracę? Mamy pewien zestaw liczb, wśród których chcemy znaleźć zarówno minimum jak i maksimum. W takim razie podzielmy wstępnie nasze liczby na kandydatów minimum oraz kandydatów maksimum. Zrobimy to przechodząc po kolei po parach sąsiednich liczb z tablicy i porównując je ze sobą. Mniejszą z wartości z pary wrzucimy do kandydatów na minimum, a większą umieścimy w kandydatach na maksimum. W ten sposób uzyskamy dwie tablice, z których każda będzie miała długość równą połowie długości pierwotnej tablicy. Teraz możemy przejść do wyszukiwania minimum i maksimum. Minimum będziemy szukać standardowym algorytmem w tablicy kandydatów na minimum. Podobnie zrobimy z maksimum, szukając go w kandydatach na maksimum.
Warning
Uwaga
Dla ułatwienia zakładamy, że długość tablicy (wartość \(n\)) jest liczbą parzystą. Jeżeli tak nie jest, możemy np. powielić ostatni element tablicy, albo rozważyć ten szczególny przypadek w algorytmie.
Pseudocode¶
funkcja SzukajMinMax(n, A):
1. kandMin := []
2. kandMax := []
3. i := 1
4. k := 1
5. Dopóki i + 1 <= n, wykonuj:
6. Jeżeli A[i] < A[i+1], to:
7. kandMin[k] := A[i]
8. kandMax[k] := A[i+1]
9. w przeciwnym przypadku:
10. kandMin[k] := A[i+1]
11. kandMax[k] := A[i]
12. k := k + 1
13. i := i + 2
14. min := kandMin[1]
15. max := kandMax[1]
16. Od i := 2 do (n div 2), wykonuj:
17. Jeżeli min > kandMin[i], to:
18. min := kandMin[i]
19. Jeżeli max < kandMax[i], to:
20. max := kandMax[i]
21. Zwróc min, max
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["SzukajMinMax(n, A)"]) --> K1["kandMin := [1 .. n div 2]
kandMax := [1 .. n div 2]
i := 1
k := 1"]
K1 --> K5{i + 1 <= n}
K5 -- TRUE --> K6{"A[i] < A[i + 1]"}
K6 -- TRUE --> K7["kandMin[k] := A[i]"]
K7 --> K8["kandMax[k] := A[i + 1]"]
K6 -- FALSE --> K10["kandMin[k] := A[i + 1]"]
K10 --> K11["kandMax[k] := A[i]"]
K8 --> K12["k := k + 1
i := i + 2"]
K11 --> K12
K12 --> K5
K5 -- FALSE --> K14["min := kandMin[1]
max := kandMax[1]
i := 1"]
K14 --> K16{i <= n div 2}
K16 -- TRUE --> K17{"min > kandMin[i]"}
K17 -- TRUE --> K18["min := kandMin[i]"]
K17 -- FALSE --> K19{"max < kandMax[i]"}
K18 --> K19
K19 -- TRUE --> K20["max := kandMax[i]"]
K19 -- FALSE --> K16i[i := i + 1]
K20 --> K16i
K16i --> K16
K16 -- FALSE ---> K21[\"Zwróc min, max"\]
K21 ----> STOP([STOP])
Complexity¶
\(O(3\frac{n}{2})\)
Najpierw dokonujemy podziału na dwie tablice pomocnicze wykonując \(\frac{n}{2}\) operacji. Następnie wyszukujemy minimum i maksimum w odpowiednich tablicach. Każda z nich ma długość \(\frac{n}{2}\), więc łącznie na znalezienie minimum i maksimum potrzebujemy wykonać \(2\frac{n}{2}=n\) porównań. Wszystko razem daje nam \(3\frac{n}{2}\) porównań. W praktyce wciąż mamy złożoność liniową, wykonujemy jednak mniej operacji niż przy algorytmie naiwnym.