Тест Миллера (теория чисел)Тест Миллера — детерминированный полиномиальный тест простоты, предложенный Миллером и впервые опубликованный в 1976 году [1]. Описание алгоритмаПринцип работыТест Миллера основывается на том, что нечётное составное число либо является степенью некоторого простого числа, либо существует простое число, лежащее в интервале от до некоторой функции , зависящей от , не являющееся свидетелем простоты данного числа по Миллеру. АлгоритмВвод: n > 2, нечётное натуральное число, которое необходимо проверить на простоту; Вывод: составное, означает, что n является составным числом; простое, означает, что n является простым числом. (1) Проверить, является ли n степенью какого-либо числа. если является, то вернуть составное (2) Найти первые m простых чисел , где m такое, что Вычислить и такие, что и - нечётное Положить перейти на шаг (4) (3) если , то если , то вернуть простое (4) если , то вернуть составное Вычислить (5) если то вернуть составное (6) если то перейти на шаг (3) Положить (7) если то перейти на шаг (3) (8) вернуть составное История и модификацииДанный тест простоты был предложен Гари Миллером в 1976 году и впервые был опубликован в журнале "Journal of Computer and System Sciences". Он основывается на работе Анкени о нахождении наименьшего невычета, опирающейся на обобщённую гипотезу Римана (для обобщений дзета-функций, называемых L-функциями Дирихле). [2]. В предположении справедливости обобщённой гипотезы Римана, на данный момент (2016 год), доказано, что в качестве функции можно взять . То есть для проверяемого на простоту числа нужно проверить, что оно сильно псевдопростое по всем простым основаниям, меньшим, , в таком случае, число - простое, если верна также обобщённая гипотеза Римана. Первоначально тот же результат был доказан для , но впоследствии в 1985 году Эрик Бах[англ.] уменьшил [3] коэффициент до 2. Однако, даже если использовать функцию , алгоритм Миллера работает на много порядков медленнее, чем вероятностный алгоритм Миллера-Рабина. Единственное преимущество этого алгоритма (Миллера), что он достоверно устанавливает простоту числа, опираясь лишь на обобщённую гипотезу Римана. А эта гипотеза хотя и не доказана до сих пор, но есть большая вероятность, а также много численных свидетельств в пользу того, что она верна. Существует также ещё более усиленная гипотеза, в пользу которой свидетельствуют некоторые теоремы и эвристические оценки. Порядок верхней оценки был позже предложен Альфордом[англ.], Гранвиллем[англ.] и Померансом. Если число , сильно псевдопростое по всем простым основаниям, меньшим чем , то число - простое. Однако, эта гипотеза на данный момент (2016 год) не доказана даже в предположении обобщённой гипотезы Римана. Возможно, позже, когда будет доказана обобщённая гипотеза Римана, и этот, ещё более сильный результат, тогда алгоритм, проверяющий простоту числа по этому условию, и будет самым быстрым доказанным, достоверным алгоритмом проверки числа на простоту. В самом деле, чтобы проверить к примеру, -битное число на простоту (то есть ), нужно убедиться всего лишь, что оно сильно псевдопростое по всем простым основаниям, меньшим чем , что совсем немного, в сравнении с оценкой , в алгоритме Миллера: мы бы проверяли по всем простым основаниям до . Алгоритм имеет полиномиальную трудоёмкость, так как при росте размера (меры) входных данных: длины записи , проверяемого числа (в любой системе счисления), вычислительная трудоёмкость растёт, со скоростью роста полинома некоторой степени от . (В случае проверки на простоту или факторизации — алгоритмы принимают только число, и размером входных данных, само число являться не может: размером данных является именно длина записи в какой-то системе счисления). Алгоритм Миллера, проверяющий по всем простым основаниям, меньшим чем , работает уже ненамного медленнее вероятностного алгоритма Миллера-Рабина, (то есть его вполне практично можно использовать), зато имеет по сравнению с ним явное преимущество. Алгоритм Миллера-Рабина — всегда явно вероятностный, то есть никогда нельзя будет узнать достоверно что любое полученное им число — простое. А данный алгоритм, скорее всего, — достоверный, только это ещё нуждается в доказательстве. (И даже если окажется, что обобщённая гипотеза Римана неверна, и тогда алгоритм Миллера будет вероятностным, но он определит простоту числа, с большей вероятностью, чем реализации алгоритма Миллера-Рабина. Потому что вероятностный алгоритм Миллера-Рабина был предложен, по сути, как упрощённый вариант алгоритма Миллера с целью повышения скорости вычислений). Нахождение именно достоверного, и вместе с тем самого быстрого алгоритма для определения простоты произвольных чисел, может быть полезно в математических теориях, в которых исследуются истинно простые числа, а не вероятностные. Вышеописанные условия проверки определены для сколь угодно больших простых чисел, а для фиксированных чисел, известны (на 2016 год), результаты, по которым числа , можно проверять на простоту, ещё быстрее. Ниже в качестве примера даны некоторые из них (для них используем тот же классический алгоритм Миллера, но с очень малым количеством оснований):
Первое составное число , которое является сильно псевдопростым по всем простым основаниям до 71 включительно, пока не найдено, но известно, что . То есть известно (на 2016 год), что все числа, меньшие чем , являющиеся сильно псевдопростыми, по первым 20-ти основаниям (до 71 включительно), являются также, простыми. Из этих данных видим, что первые натуральных чисел, можно проверить на простоту (причём, достоверно, так как это уже доказано), используя количество оснований, даже меньше чем во всех выше полученных оценках. Этот алгоритм - самый быстрый для проверки чисел на простоту. Обратное же утверждение следующее — если число простое, то оно сильно псевдопростое, вообще по всем основаниям. Также есть версия алгоритма, не использующая расширенную гипотезу Римана. Этот алгоритм имеет экспоненциальную вычислительную сложность. В таком случае в роли функции выступает функция . Свойства
Время работыВ случае, когда верхняя граница перебора задаётся функцией , алгоритм детерминировано проверяет простоту числа n за [4] , но при этом алгоритм опирается на обобщённую гипотезу Римана. Проще говоря, трудоёмкость алгоритма растёт быстрее чем , (количество проверяемых оснований на псевдопростоту), потому что чем больше основание, тем более трудоёмкий его анализ. От размера (меры) входных данных: длины записи , проверяемого числа , трудоёмкость алгоритма значит, зависит так: , то есть полиномиальная сложность, 4-й степени. В случае, когда , время работы в худшем случае оценивается как [4]. От размера входных данных: длины записи , проверяемого числа , трудоёмкость алгоритма, зависит так: , то есть экспоненциальная сложность степени 1/7. Этот алгоритм намного сложнее: все экспоненциальные алгоритмы, при достаточно большом размере входных данных, становятся существенно более трудоёмкими, чем все полиномиальные алгоритмы. Пример реализации алгоритмаПример реализации алгоритма, на языке программирования C# (.NET Framework 3.5, 4.0). Это только один из примеров, и переменную maxChecked, можно определить по другому, и по формуле от проверяемого числа (классический тест Миллера), и по точным значениям для чисел, меньших чем , описанным выше, и вообще произвольным образом так, что получится реализация алгоритма Миллера-Рабина. public bool IsPrime_AlgMiller(BigInteger checkingNum)
{
// (если является степенью другого числа)
if (IsPowerOfNumber(checkingNum))
return false;
BigInteger logN = new BigInteger(BigInteger.Log(checkingNum));
BigInteger loglogN = new BigInteger(BigInteger.Log(logN));
BigInteger maxChecked = logN * loglogN / (new BigInteger(BigInteger.Log(2)));
// maxChecked: получено максимальное основание, для проверки на сильную псевддопростоту. Это один из примеров.
BigInteger baseCurrent = new BigInteger(2);
bool isPrime = true;
while (baseCurrent <= maxChecked)
{
// (если не сильно псевдопростое по этому основанию)
if (! IsStrongPseudoPrime(checkingNum, baseCurrent))
{
// (тогда число не простое)
isPrime = false;
break;
}
baseCurrent = NextPrime(baseCurrent);
}
return isPrime;
}
public bool IsStrongPseudoPrime(BigInteger checkingNum, BigInteger baseCurrent)
{
BigInteger exp = checkingNum - 1;
// (exp будет меняться, а проверка остатка -1 эквивалентна проверке остатка (checkingNum - 1))
BigInteger ost_1 = exp;
BigInteger res = BigInteger.ModPow(baseCurrent, exp, checkingNum);
if (res != 1)
return false;
// (тест Ферма пройден)
while (true)
{
// (чётное; при первом попадании всегда будет чётным, далее цикл до тех пор пока снова станет нечётным)
exp = exp / 2;
// (остаток -1 всегда должны проверить)
res = BigInteger.ModPow(baseCurrent, exp, checkingNum);
if (res == ost_1)
return true;
// (снова стало нечётным — нужно проверить ещё на 1)
if (!exp.IsEven)
{
res = BigInteger.ModPow(baseCurrent, exp, checkingNum);
if (res.IsOne)
return true;
break;
}
}
return false;
}
public BigInteger NextPrime(BigInteger num)
{
//...
}
public bool IsPowerOfNumber(BigInteger n)
// n=p^k => p-1|n-1 && 2^(p-1)=1(mod p) => 2^(n-1)=1(mod p) => p|2^(n-1)-1 => p|gcd(2^(n-1)-1,n)
{
return BigInteger.GreatestCommonDivisor(BigInteger.ModPow(2, n - 1, n)-1, n)>1;
}
Остаётся реализовать только две функции - 1) NextPrime, которая получает число, и возвращает следующее за ним, простое число, оптимальная для нахождения именно малых простых чисел. Эта функция должна быть ещё проще и быстрее. 2) IsPowerOfNumber - функция, немного более сложная, которая определяет, является ли передаваемое число, степенью другого, простого числа. Нужно найти максимально простую реализацию этой функции. Также, 3) Можно ускорить реализацию алгоритма, отсеяв вначале, случаи, когда число делится на возможные малые делители, но часто-встречающиеся делители: 2,3,5,7,11... и только потом выполнять основную проверку по алгоритму Миллера. В таком случае реализация алгоритма Миллера в предложенном примере, скорее всего, будет наибыстрейшей, для произвольных больших чисел. А сам алгоритм, как уже показано выше, претендует на звание самого быстрого достоверного алгоритма на проверку простоты (если верна обобщённая гипотеза Римана). Эта реализация алгоритма была успешно протестирована и без использования функции IsPowerOfNumber. ПримечанияЛитература
|