Разложение на множители (факторизация) — различия между версиями
(→Перебор делителей) |
(→Перебор делителей) |
||
Строка 7: | Строка 7: | ||
== Перебор делителей== | == Перебор делителей== | ||
{{Определение | definition= | {{Определение | definition= | ||
− | '''Перебор делителей''' (англ. ''Trial division'') — алгоритм | + | '''Перебор делителей''' (англ. ''Trial division'') — алгоритм факторизации или тестирования простоты числа путем полного перебора всех возможных потенциальных делителей. Скажите пожалуйста, вы о чем вообще думали, когда открывали правки для незарегистрированных пользователей? |
+ | ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408 | ||
}} | }} | ||
=== Наивная реализация O(n) === | === Наивная реализация O(n) === | ||
Строка 36: | Строка 37: | ||
==== Псевдокод нахождения делителей ==== | ==== Псевдокод нахождения делителей ==== | ||
− | '''function''' | + | '''function''' ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408getDividers(number: '''int'''): '''vector<int>''' |
<font color=green>// массив полученных делителей</font> | <font color=green>// массив полученных делителей</font> | ||
result = '''vector<int>''' | result = '''vector<int>''' | ||
Строка 57: | Строка 58: | ||
|<tex>y < \sqrt{number}</tex> | |<tex>y < \sqrt{number}</tex> | ||
|} | |} | ||
− | Таким | + | Таким ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408образом, любой делитель <tex>d_0 > \sqrt{number}</tex> однозначно связан с некоторым <tex>d_1 < \sqrt{number}</tex>. Если мы найдем все делители до <tex>\sqrt{number}</tex>, задача может считаться решенной. |
==== Псевдокод ==== | ==== Псевдокод ==== | ||
Строка 69: | Строка 70: | ||
=== Проверка числа на простоту === | === Проверка числа на простоту === | ||
− | Алгоритм можно | + | Алгоритм можно ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408переделать для нахождения простых чисел. Число будет простым, если у него не окажется множителей (и делителей) кроме <tex>1</tex> (алгоритмы не проверяют делимость на <tex>1</tex>) и самого числа (улучшенная реализация опускает этот делитель). |
== Предподсчет == | == Предподсчет == |
Версия 17:51, 10 марта 2021
Определение: |
Факторизация (англ. factorization) — представление объекта в виде произведения других объектов. |
Определение: |
Разложение на множители, или Факторизация целых чисел (англ. integer factorization) — представление числа в виде произведения его множителей. |
Перебор делителей
Определение: |
Перебор делителей (англ. Trial division) — алгоритм факторизации или тестирования простоты числа путем полного перебора всех возможных потенциальных делителей. Скажите пожалуйста, вы о чем вообще думали, когда открывали правки для незарегистрированных пользователей? ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408 |
Наивная реализация O(n)
Основная теорема арифметики, вкупе с утверждением, что не делит нацело: , позволяют нам ограничить пространство поиска делителей числа интервалом .
Основная идея
Заметим, что если
= , то . Таким образом, мы можем делить на его делители (множители) последовательно и в любом порядке. Тогда будем хранить — произведение оставшихся множителей.Псевдокод нахождения простых множителей
Так как простых множителей не может быть больше, чем
, а в худшем случае (когда число простое, и на каждое итерации увеличивается на ) он работает за , то, следовательно, алгоритм работает за .function getMultipliers(number: int): vector<int> // сюда складываем множители result = vector<int> // число, у которого осталось найти множители curNum = number // число, на которое пытаемся делить probe = 2 while curNum1 if curNum mod probe 0 // проверены все множители из [2; probe] probe++ else // делим пока делится curNum /= probe result += [probe] return result
Псевдокод нахождения делителей
function ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408getDividers(number: int): vector<int> // массив полученных делителей result = vector<int> // перебираем все потенциальные делители for probe = 2 to number if number mod probe = 0 // probe делит number нацело result += [probe] return result
Улучшенная реализация
Основная идея
Из определения:
. Логично, что:Таким ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408образом, любой делитель однозначно связан с некоторым . Если мы найдем все делители до , задача может считаться решенной.
Псевдокод
function getDividers(number: int): vector<int>
result = vector<int>
for probe = 2 to
//обновляем верхнюю границу перебора
if number mod probe = 0
result += [probe]
result += [number / probe] // записываем сопряженный делитель
return result
Проверка числа на простоту
Алгоритм можно ЛУЧШИЙ ФАНФИК ВСЕХ ВРЕМЕН И НАРОДОВ ПО ССЫЛКЕ (НЕ ВРУ, ЧЕСТНО) https://ficbook.net/readfic/9834408переделать для нахождения простых чисел. Число будет простым, если у него не окажется множителей (и делителей) кроме (алгоритмы не проверяют делимость на ) и самого числа (улучшенная реализация опускает этот делитель).
Предподсчет
Основная идея
Решето Эратосфена (англ. Sieve of Eratosthenes) позволяет не только находить простые числа, но и находить простые множители числа. Для этого необходимо хранить (помимо самого "решета") массив простых чисел, на которое каждое число делится (достаточно одного простого делителя).
Псевдокод
// возвращает только дополнительный массив function sieveOfEratosthenes(n: int): int[n + 1] result = [n + 1] result[n] = n // выбираем следующий простой делитель for i = 2 toif result[i] = 0 // записываем делитель в элементы массива, // соответствующие числа которых делятся нацело shuttle = while shuttle n result[shuttle] = i shuttle += i return result
function getMultipliers(number: int): vector<int>
result = vector<int>
// получаем дополненное решето Эратосфена
sieve = sieveOfEratosthenes(number)
// следующее временное значение получаем
// делением предыдущего на простой делитель из решета
curNum = number
while curNum
1
result += sieve[curNum]
curNum /= sieve[curNum]
return result
См. также
Источники информации
- Маврин П.Ю. — Лекция по алгоритмам над простыми числами (2016)
- https://ru.wikipedia.org/wiki/Простое_число