Test pierwszości¶
Liczby pierwsze odgrywają w matematyce i informatyce znaczącą rolę, szczególnie w kryptografii. Zacznijmy od zdefiniowania czym jest liczba pierwsza:
Info
Liczba pierwsza
Liczbę naturalną, która jest podzielna wyłącznie przez \(1\) i przez siebie samą, nazywamy liczbą pierwszą.
Naszym zadaniem jest sprawdzić, czy dana liczba naturalna jest liczbą pierwszą. Jednym ze sposobów jest sprawdzić, jakie dzielniki ma ta liczba. Nasze podejście do tego problemu będzie bardzo podobne jak w przypadku wypisania wszystkich dzielników liczby, dlatego warto najpierw zajrzeć do tego właśnie tematu: Wszystkie dzielniki
Jak zwykle zaczynamy od formalnej specyfikacji problemu i kilku prostych przykładów.
Specification¶
Input¶
- \(n\) - liczba naturalna większa od \(1\)
Output¶
- TRUE- jeżeli \(n\) jest liczbą pierwszą
- FALSE- jeżeli \(n\) nie jest liczbą pierwszą
Info
Dlaczego w specyfikacji wymagamy, żeby liczba \(n\) była większa od \(1\)? Dla ułatwienia i zwiększenia czytelności rozwiązania. Wiadomo, że liczby \(0\) i \(1\) nie są liczbami pierwszymi. W związku z tym możemy je łatwo potraktować jako osobny przypadek, używając instrukcji warunkowej. To jednak zostawimy jako ćwiczenie do samodzielnego wykonania.
Example 1¶
Input¶
Output: TRUE
Info
Wyjaśnienie
Dzielnikami liczby \(7\) są \(1\) i \(7\), więc jest to liczba pierwsza.
Example 2¶
Input¶
Output: FALSE
Info
Wyjaśnienie
Dzielnikami liczby 8 są \(1,2,4\) i \(8\), więc nie jest to liczba pierwsza.
Solution zupełnie naiwne¶
W pierwszym rozwiązaniu postępować będziemy podobnie, jak w przypadku wypisywania wszystkich dzielników liczby. Są jednak dwie znaczące różnice. Po pierwsze nie interesuje nas, czy liczba jest podzielna przez \(1\) i samą siebie, ponieważ z góry wiemy, że tak jest. Możemy więc nieznacznie zawęzić obszar poszukiwań. Po drugie nie interesuje nas jakie dokładnie dzielniki ma liczba, tylko czy ma jakieś dzielniki, różne od \(1\) i niej samej. W związku z tym, jak tylko znajdziemy jakiś dzielnik, możemy od razu stwierdzić, że liczba nie jest pierwsza i zwrócić właściwą wartość, czyli FALSE i zakończyć obliczenia.
Jak jednak sprawdzić, że liczba jest pierwsza? To proste. Wystarczy, że nie znajdziemy żadnych dzielników. Jak nie znajdziemy dzielników, to nie zwrócimy wartości FALSE, tylko wyjdziemy z pętli. Jeśli tak się stanie, to znaczy, że liczba jest pierwsza, należy więc zwrócić wartość TRUE.
Podsumujmy nasze rozważania w formie gotowego algorytmu.
Pseudocode¶
funkcja CzyPierwsza(n):
1. Od i := 2 do n - 1, wykonuj:
2. Jeżeli (n mod i) = 0, to:
3. Zwróć FALSE
4. Zwróć TRUE
Info
mod oznacza operację reszty z dzielenia
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["CzyPierwsza(n)"]) --> K0[i := 2]
K0 --> K1{i < n}
K1 -- TRUE --> K2{n mod i = 0}
K2 -- TRUE --> K3[/Zwróć FALSE/]
K3 --> STOP([STOP])
K2 -- FALSE --> K1i[i := i + 1]
K1i --> K1
K1 -- FALSE ----> K4[/Zwróć TRUE/]
K4 --> STOP
Wizualizacja dla n = 10¶
Wizualizacja testu pierwszości
Complexity¶
\(O(n)\) - liniowa
Solution naiwne¶
Mamy już pierwsze rozwiązanie naszego problemu. Zastanówmy się teraz, jak możemy je zoptymalizować, czyli usprawnić. Szczególnym fragmentem naszego rozwiązania, który aż prosi się o optymalizację, jest przeglądanie liczb od \(2\) do \(n-1\). Pomyślmy, jak możemy zawęzić ten zakres?
Zauważmy, że od pewnej wartości możemy mieć już gwarancję, że nie znajdziemy kolejnych dzielników. Gdy sprawdzana liczba jest większa od połowy \(n\), to nie może być już dzielnikiem \(n\). W związku z tym wystarczy, że będziemy sprawdzać potencjalne dzielniki do \(n/2\), a dokładniej do części całkowitej z tegoż dzielenia.
Pseudocode¶
funkcja CzyPierwsza(n):
1. Od i := 2 do n div 2, wykonuj:
2. Jeżeli (n mod i) = 0, to:
3. Zwróć FALSE
4. Zwróć TRUE
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["CzyPierwsza(n)"]) --> K0[i := 2]
K0 --> K1{i <= n div 2}
K1 -- TRUE --> K2{n mod i = 0}
K2 -- TRUE --> K3[/Zwróć FALSE/]
K3 --> STOP([STOP])
K2 -- FALSE --> K1i[i := i + 1]
K1i --> K1
K1 -- FALSE ----> K4[/Zwróć TRUE/]
K4 --> STOP
Complexity¶
\(O(\frac{n}{2})\)
Solution optymalne¶
Możemy jeszcze bardziej skrócić zakres przeszukiwanych wartości. Można matematycznie udowodnić, że jeżeli liczba naturalna jest liczbą złożoną, to posiada przynajmniej jeden dzielnik (poza jedynką) mniejszy bądź równy pierwiastkowi z siebie samej.
Dlatego możemy naszą pętlę ograniczyć do pierwiastka z \(n\) włącznie.
Pseudocode¶
funkcja CzyPierwsza(n):
1. Od i := 2 do sqrt(n), wykonuj:
2. Jeżeli (n mod i) = 0, to:
3. Zwróć FALSE
4. Zwróć TRUE
Info
sqrt oznacza pierwiastek
Block diagram¶
%%{init: {"flowchart": {"curve": "linear"}, "theme": "neutral"} }%%
flowchart TD
START(["CzyPierwsza(n)"]) --> K0[i := 2]
K0 --> K1{"i <= sqrt(n)"}
K1 -- TRUE --> K2{n mod i = 0}
K2 -- TRUE --> K3[/Zwróć FALSE/]
K3 --> STOP([STOP])
K2 -- FALSE --> K1i[i := i + 1]
K1i --> K1
K1 -- FALSE ----> K4[/Zwróć TRUE/]
K4 --> STOP
Complexity¶
\(O(\sqrt{n})\)