Algorytm Euklidesa¶
Algorytm Euklidesa to jeden z najstarszych znanych algorytmów, który został opisany przez starożytnego greckiego matematyka Euklidesa w jego dziele "Elementy" około 300 roku p.n.e. Algorytm ten służy do znajdowania największego wspólnego dzielnika (NWD) dwóch liczb naturalnych. Jest on oparty na prostej obserwacji, że NWD dwóch liczb nie zmienia się, jeśli większą z tych liczb zastąpimy różnicą tych liczb. Dzięki temu możemy iteracyjnie zmniejszać wartości liczb, aż do momentu, gdy staną się one równe, co oznacza, że znaleźliśmy NWD.
Algorytm Euklidesa znajduje zastosowanie w wielu dziedzinach matematyki i informatyki, m.in. w kryptografii (np. w algorytmie RSA), w teorii liczb, a także w problemach związanych z obliczeniami na wielkich liczbach. Jego prostota i efektywność sprawiają, że jest to jedno z podstawowych narzędzi w arsenale każdego matematyka i programisty.
Specification¶
Input¶
- \(a, b\) — liczby naturalne, większe od zera, tzn. \(a,b\in\mathbb{N}\), \(a>0,b>0\)
Output¶
- \(\mathrm{NWD}(a, b)\) — największy wspólny dzielnik liczb \(a\) i \(b\)
Example¶
Input¶
Output¶
\(\mathrm{NWD}(32, 12) = 4\)
Info
Wyjaśnienie
Dzielnikami liczby \(32\) są: \(1, 2, 4, 8, 16, 32\)
Dzielnikami liczby \(12\) są: 1, 2, 3, 4, 6, 12
Wspólnymi dzielnikami są więc: \(1, 2, 4\)
Największy z nich to właśnie \(4\).
Wersja z odejmowaniem¶
Zasada jest prosta: od większej liczby odejmujemy mniejszą i tak w kółko, aż uzyskamy dwie takie same wartości, które będą naszym wynikiem. Proces ten można opisać następująco:
- Porównujemy dwie liczby, \(a\) i \(b\).
- Jeśli \(a\) jest większe od \(b\), odejmujemy \(b\) od \(a\).
- Jeśli \(b\) jest większe od \(a\), odejmujemy \(a\) od \(b\).
- Powtarzamy kroki 1-3, aż obie liczby będą równe.
- Gdy obie liczby są równe, osiągnęliśmy największy wspólny dzielnik (NWD).
Example 1¶
a | b |
---|---|
28 | 12 |
28-12=16 | 12 |
16-12=4 | 12 |
4 | 12-4=8 |
4 | 8-4=4 |
\(\mathrm{NWD}(28, 12)=4\)
Example 2¶
a | b |
---|---|
3 | 16 |
3 | 13 |
3 | 10 |
3 | 7 |
3 | 4 |
3 | 1 |
2 | 1 |
1 | 1 |
\(\mathrm{NWD}(3, 16) = 1\)
Example 3¶
a | b |
---|---|
6 | 18 |
6 | 12 |
6 | 6 |
\(\mathrm{NWD}(6,18)=6\)
Pseudocode¶
funkcja NWD(a, b):
1. Dopóki a != b, wykonuj:
2. Jeżeli a > b, to:
3. a := a - b
4. W przeciwnym przypadku:
5. b := b - a
6. Zwróć a
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["NWD(a, b)"]) --> K1{a != b}
K1 -- TRUE --> K2{a > b}
K2 -- TRUE --> K3[a := a - b]
K3 --> K1
K2 -- FALSE --> K5[b := b - a]
K5 --> K1
K1 -- FALSE --> K6[/Zwróć a/]
K6 --> STOP([STOP])
Wersja z modulo — iteracyjna¶
Odejmowanie możemy zastąpić operacją reszty z dzielenia, która jest dużo wydajniejsza w tym przypadku.
Example 1¶
a | b |
---|---|
28 | 12 |
12 (b z wiersza wyżej) | 28 mod 12 (a % b) = 4 |
4 (b z wiersza wyżej) | 12 mod 4 (a % b) = 0 |
\(\mathrm{NWD}(28, 12)=4\)
Example 2¶
a | b |
---|---|
3 | 16 |
16 | 3 mod 16 = 3 |
3 | 16 mod 3 = 1 |
1 | 3 mod 1 = 0 |
\(\mathrm{NWD}(3, 16) = 1\)
Example 3¶
a | b |
---|---|
6 | 18 |
18 | 6 |
6 | 0 |
\(\mathrm{NWD}(6,18)=6\)
Example 4¶
a | b |
---|---|
100 | 2 |
2 | 0 |
\(\mathrm{NWD}(100, 2) = 2\)
Pseudocode¶
Info
mod oznacza resztę z dzielenia
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["NWD(a, b)"]) --> K1{b != 0}
K1 -- TRUE --> K2["b2 := b
b := a mod b
a := b2"]
K2 --> K1
K1 -- FALSE --> K5[/Zwróć a/]
K5 --> STOP([STOP])
Wersja z modulo — rekurencyjna¶
Opisaną wyżej metodę możemy również zdefiniować rekurencyjnie.
Pseudocode¶
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["NWD(a, b)"]) --> K1{b = 0}
K1 -- TRUE --> K2[/Zwróć a/]
K2 --> STOP([STOP])
K1 -- FALSE --> K3[/"Zwróć NWD(b, a mod b)"/]
K3 --> STOP