Русская Википедия:Решето Эратосфена

Материал из Онлайн справочника
Версия от 08:19, 11 сентября 2023; EducationBot (обсуждение | вклад) (Новая страница: «{{Русская Википедия/Панель перехода}} '''Решето́ Эратосфе́на''' — алгоритм нахождения всех простых чисел до некоторого целого числа {{math|''n''}}, который приписывают древнегреческому математику Эратосфену Киренскому<ref name="nicom...»)
(разн.) ← Предыдущая версия | Текущая версия (разн.) | Следующая версия → (разн.)
Перейти к навигацииПерейти к поиску

Решето́ Эратосфе́на — алгоритм нахождения всех простых чисел до некоторого целого числа Шаблон:Math, который приписывают древнегреческому математику Эратосфену Киренскому[1]. Как и во многих случаях, здесь название алгоритма говорит о принципе его работы, то есть решето подразумевает фильтрацию, в данном случае фильтрацию всех чисел за исключением простых. По мере прохождения списка нужные числа остаются, а ненужные (они называются составными) исключаются.

История

Этот метод описан во «Введении в арифметику» Никомаха Герасского. Никомах называет автором метода Эратосфена. То же делает и Ямвлих в своём комментарии к этому сочинению Никомаха.

Название «решето» метод получил потому, что во времена Эратосфена писали числа на дощечке, покрытой воском, и прокалывали дырочки в тех местах, где были написаны составные числа. Поэтому дощечка являлась неким подобием решета, через которое «просеивались» все составные числа, а оставались только числа простые[2].

Никомах,[1] во 2-м веке н.э., объясняет, что метод решета "высеивает" простые числа из нечётных, отделяя от них составные числа, которые он находит, перечисляя для каждого нечетного числа n каждое n-ное число в ряду нечётных чисел, начиная с n. Символически,

 Primes = {3,5,7,9,...} \ Composites
 Composites = { 3n,5n,7n,9n,... for n in {3,5,7,9,...} }

Британец Хорслей,[3] 16 веков спустя, горячо критикует это описание, заявляя что "истинный" метод Эратосфена "наверняка" был "гораздо умнее", начиная с квадратов простых чисел прямо в процессе их распознавания:

 Composites = { ,+2n,+4n,... for n in Primes }

Алгоритм

Анимация шагов алгоритма Эратосфена для нахождения простых чисел до 120
Анимация шагов алгоритма Эратосфена для нахождения простых чисел до 120

Для нахождения всех простых чисел не больше заданного числа n, следуя методу Эратосфена, нужно выполнить следующие шаги:

  1. Выписать подряд все целые числа от двух до n (2, 3, 4, …, n).
  2. Пусть переменная p изначально равна двум — первому простому числу.
  3. Зачеркнуть в списке числа от 2p до n, считая шагами по p (это будут числа, кратные p: 2p, 3p, 4p, …).
  4. Найти первое незачёркнутое число в списке, большее чем p, и присвоить значению переменной p это число.
  5. Повторять шаги 3 и 4, пока возможно.

Теперь все незачёркнутые числа в списке — это все простые числа от 2 до n.

На практике, алгоритм можно улучшить следующим образом. На шаге № 3 числа можно зачеркивать, начиная сразу с числа p2, потому что все меньшие числа, кратные p, обязательно имеют простой делитель меньше p, а они уже зачеркнуты к этому времени. И, соответственно, останавливать алгоритм можно, когда p2 станет больше, чем n.[3] Кроме того, все простые числа, кроме 2, — нечётные числа, и поэтому для них можно считать шагами по 2p, начиная с p2.

Пример для n = 30

Запишем натуральные числа, начиная от Шаблон:Math, до Шаблон:Math в ряд:

2  3  4  5  6  7  8  9  10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30

Первое число в списке, Шаблон:Math — простое. Пройдём по ряду чисел, зачёркивая все числа, кратные Шаблон:Math (то есть, каждое второе, начиная с Шаблон:Math):

2  3  4  5  6  7  8  9  10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30

Следующее незачеркнутое число, Шаблон:Math — простое. Пройдём по ряду чисел, зачёркивая все числа, кратные Шаблон:Math (то есть, каждое третье, начиная с Шаблон:Math):

2  3  4  5  6  7  8  9  10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30

Следующее незачеркнутое число, Шаблон:Math — простое. Пройдём по ряду чисел, зачёркивая все числа, кратные 5 (то есть, каждое пятое, начиная с Шаблон:Math):

2  3  4  5  6  7  8  9  10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30

Следующее незачеркнутое число — Шаблон:Math. Его квадрат, Шаблон:Math — больше Шаблон:Math, поэтому на этом работа завершена. Все составные числа уже зачеркнуты:

2  3     5     7           11    13          17    19          23                29

Псевдокод

Оптимизированная реализация (начинающаяся с квадратов) на псевдокоде[4][5]:

Вход: натуральное число n
Выход: все простые числа от 2 до n.

Пусть Aбулевый массив, индексируемый числами от 2 до n, 
изначально заполненный значениями true.

 для i = 2, 3, 4, ..., пока i2n:
  если A[i] = true:
    для j = i2, i2 + i, i2 + 2i, ..., пока jn:
      назначить A[j] := false

 возвращаем: все числа i, для которых A[i] = true.

Сложность алгоритма

Сложность алгоритма составляет <math>O(n \log (\log n))</math> операций при составлении таблицы простых чисел до <math>n</math>[6].

Доказательство сложности

При выбранном <math>n \in \mathbb{N}</math> для каждого простого <math>p \in \mathbb{P}\colon p \le n</math> будет выполняться внутренний цикл[7], который совершит <math>\frac{n}{p}</math> действий. Сложность алгоритма можно получить из оценки следующей величины:

<math> \sum\limits_{p\in \mathbb{P}\colon p \le n} {\frac{n}{p}} = n\cdot\sum\limits_{p\in \mathbb{P}\colon p \le n} {\frac{1}{p}} </math>

Так как количество простых чисел, меньших либо равных <math>n</math>, оценивается как <math>\frac{n}{\ln n}</math>, и, как следствие, <math>k</math>-е простое число примерно равно <math>k\ln k</math>, то сумму можно преобразовать:

<math> \sum\limits_{p \in \mathbb{P}\colon p \le n}\frac{1}{p}\approx \frac{1}{2} + \sum\limits_{k=2}^{\frac{n}{\ln n}}\frac{1}{k\ln k}</math>

Здесь из суммы выделено слагаемое для первого простого числа, чтобы избежать деления на ноль. Данную сумму можно оценить интегралом

<math> \frac{1}{2} + \sum^{\frac{n}{\ln n}}_{k=2}{\frac{1}{k\ln k}} \approx \int\limits_2^{\frac{n}{\ln n}} {\frac{1}{k\ln k}}\,dk = \ln \ln k \Bigr|_2^{\frac{n}{\ln n}} = \ln \ln {\frac{n}{\ln n}} - \ln \ln 2 = \ln (\ln n - \ln \ln n) - \ln \ln 2 \approx \ln \ln n </math>

В итоге получается для изначальной суммы:

<math> \sum\limits_{p\in \mathbb{P}\colon p \le n} {\frac{n}{p}} \approx n \ln \ln n + o(n)</math>

Более строгое доказательство (и дающее более точную оценку с точностью до константных множителей) можно найти в книге Hardy и Wright «An Introduction to the Theory of Numbers»[8].

Модификации метода

Неограниченный, постепенный вариант

В этом варианте простые числа вычисляются последовательно, без ограничения сверху,[9] как числа, находящиеся в промежутках между составными числами, которые вычисляются для каждого простого числа Шаблон:Math, начиная с его квадрата, с шагом в Шаблон:Math (или для нечетных простых чисел Шаблон:Math)[3]. Может быть представлен абстрактно в парадигме потоков данных как

 primes = {2,3,...} \ { , +p, ... for p in primes }

используя нотацию абстракции списков, где \ обозначает разницу между арифметическими прогрессиями.

Первое простое число Шаблон:Math (среди возрастающих положительных целых чисел) заранее известно, поэтому в этом самореферентном определении нет порочного круга.

Более конкретный псевдокод с поэтапным отсеиванием, в неэффективной реализации, для простоты сравнения с нижеследующими вариантами:

 primes = sieve [2..] where
    sieve [p, ...xs] = [p, ...sieve (xs \ [, +p..])]

Более эффективный вариант отделяет на каждом шагу из начала списка не одно лишь первое число, а сразу все числа не превосходящие квадрата очередного простого числа. Это реализуется посредством откладывания отсеивания каждым простым числом, до его квадрата:

 primes = [2, ...sieve primes [3..]] where
    sieve [p, ...ps] [...h, , ...xs]
         = [...h, ...sieve ps (xs \ [, +p..])]

Перебор делителей

Решето Эратосфена часто путают с алгоритмами, которые поэтапно Шаблон:Iw составные числа, тестируя каждое из чисел-кандидатов на делимость по одному простому числу на каждом этапе.[10]

Широко известный функциональный код Дэвида Тёрнера 1975 г.[11] часто принимают за решето Эратосфена,[10] но на самом деле[9] это неоптимальный вариант с перебором делителей:

 primes = sieve [2..] where
    sieve [p, ...xs] = [p, ...sieve [x in xs if x%p > 0]]

В оптимальном варианте не используются делители, большие квадратного корня тестируемого числа.

Сегментированное решето

Как замечает Соренсон, главной проблемой реализации решета Эратосфена на вычислительных машинах является не количество выполняемых операций, а требования по объёму занимаемой памяти (впрочем, его замечание относится к неактуальному сейчас компьютеру DEC VAXstation 3200, выпущенному в 1989 году).Шаблон:R При больших значениях Шаблон:Mvar, диапазон простых чисел может превысить доступную память; хуже того, даже при сравнительно небольших Шаблон:Mvar использование кэша памяти далеко от оптимального, так как алгоритм, проходя по всему массиву, нарушает принцип Шаблон:Iw.

Для решения представленной проблемы используется сегментированное решето, в котором за итерацию просеивается только часть диапазона простых чисел.[12] Данный метод известен с 1970-х годов и работает следующим образом:Шаблон:R[13]

  1. Разделяем диапазон от 2 до Шаблон:Mvar на отрезки (сегменты) некоторой длины Шаблон:Math.
  2. Находим все простые числа в первом отрезке, используя обычное решето.
  3. Каждый из последующих отрезков оканчивается на некотором числе Шаблон:Mvar. Находим все простые числа в отрезке следующим образом:
    1. Создаем булевый массив размера Шаблон:Math
    2. Для каждого простого числа Шаблон:Math из уже найденных, отмечаем в массиве как «непростые» все числа кратные Шаблон:Math, перебирая числа с шагом в Шаблон:Math, начиная с наименьшего кратного Шаблон:Math числа в данном отрезке.

Если число Шаблон:Math выбрано равным Шаблон:Math, то сложность алгоритма по памяти оценивается как Шаблон:Math, а операционная сложность остаётся такой же, что и у обычного решета Эратосфена.[13]

Для случаев, когда Шаблон:Mvar настолько велико, что все просеиваемые простые числа меньшие Шаблон:Math не могут уместиться в памяти, как того требует алгоритм сегментированного сита, используют более медленные, но с гораздо меньшими требованиями по памяти алгоритмы, например решето Соренсона.[14]

Решето Эйлера

Доказательство тождества Эйлера для дзета-функции Римана содержит механизм отсеивания составных чисел подобный решету Эратосфена, но так, что каждое составное число удаляется из списка только один раз. Схожее решето описано в Gries & Misra 1978 г. как решето с линейным временем работы (см. ниже).

Составляется исходный список начиная с числа Шаблон:Math. На каждом этапе алгоритма первый номер в списке берется как следующее простое число, результаты произведения которого на каждое число в списке помечаются для последующего удаления. После этого из списка убирают первое число и все помеченные числа, и процесс повторяется вновь:

[2] (3) 5  7  9 11  13 15 17 19 21 23 25 27 29 31 33 35 37 39 41 43 45 47 49 51 53 55 57 59 61 63 65 67 69 71 73 75 77 79  ...
[3]    (5) 7    11  13    17 19    23 25    29 31    35 37    41 43    47 49    53 55    59 61    65 67    71 73    77 79  ...
[4]       (7)   11  13    17 19    23       29 31       37    41 43    47 49    53       59 61       67    71 73    77 79  ...
[5]            (11) 13    17 19    23       29 31       37    41 43    47       53       59 61       67    71 73       79  ...
[...] 

Здесь показан пример начиная с нечетных чисел, после первого этапа алгоритма. Таким образом, после Шаблон:Math-го этапа рабочий список содержит только числа взаимно простые с первыми Шаблон:Math простыми числами (то есть числа не кратные ни одному из первых Шаблон:Math простых чисел), и начинается с Шаблон:Math-го простого числа. Все числа в списке, меньшие квадрата его первого числа, являются простыми.

В псевдокоде,

 primes = sieve [2..] where
    sieve [p, ...xs] = [p, ...sieve (xs \ [, ...p*xs])]

Решето только по нечётным числам

Поскольку все чётные числа, кроме 2, — составные, то можно вообще не обрабатывать никак чётные числа, а оперировать только нечётными числами. Во-первых, это позволит вдвое сократить объём требуемой памяти. Во-вторых, это уменьшит количество выполняемых алгоритмом операций (примерно вдвое).

В псевдокоде:

 primes = [2, ...sieve [3,5..]] where
    sieve [p, ...xs] = [p, ...sieve (xs \ [, +2p..])]

Это можно обобщить на числа взаимно простые не только с 2 (то есть нечетные числа), но также и с 3, 5, и т. д..

Уменьшение объёма потребляемой памяти

Алгоритм Эратосфена фактически оперирует с <math>n</math> битами памяти. Следовательно, можно существенно сэкономить потребление памяти, храня <math>n</math> переменных булевского типа не как <math>n</math> байт, а как <math>n</math> бит, то есть <math>n/8</math> байт памяти.

Такой подход — «битовое сжатие» — усложняет оперирование этими битами. Любое чтение или запись бита будут представлять собой несколько арифметических операций. Но с другой стороны существенно улучшается компактность в памяти. Бо́льшие интервалы умещаются в кэш-память которая работает гораздо быстрее обычной так что при работе по-сегментно общая скорость увеличивается.

Решето Эратосфена с линейным временем работы

Нижеследующий алгоритм находит составные числа как перечисление формулы

{ (piqjrk...) for p,q,r,... in primes, where i+j+k+... > 1 }

Этот алгоритм обнаруживает для каждого числа Шаблон:Math в отрезке Шаблон:Math его минимальный простой делитель Шаблон:Math (Шаблон:Math от Шаблон:Lang-en).

Также поддерживается список всех простых чисел — массив Шаблон:Math, поначалу пустой. В ходе работы алгоритма этот массив будет постепенно заполняться.

Изначально все величины Шаблон:Math заполним нулями.

Дальше следует перебрать текущее число Шаблон:Math от Шаблон:Math до Шаблон:Math. Здесь может быть два случая:

  • Шаблон:Math: это означает, что число Шаблон:Math — простое, так как для него так и не обнаружилось других делителей.

Следовательно, надо присвоить Шаблон:Math и добавить Шаблон:Math в конец списка Шаблон:Math.

В обоих случаях дальше начинается процесс расстановки значений в массиве Шаблон:Math: следует брать числа, кратные Шаблон:Math, и обновлять у них значение Шаблон:Math. Однако основная цель — научиться делать это таким образом, чтобы в итоге у каждого числа значение Шаблон:Math было бы установлено не более одного раза.

Утверждается, что для этого можно поступить таким образом. Рассматриваются числа вида Шаблон:Math, где Шаблон:Math последовательно равно всем простым числам не превосходящим Шаблон:Math (как раз для этого понадобилось хранить список всех простых чисел).

У всех чисел такого вида проставим новое значение Шаблон:Math — оно должно быть равно Шаблон:Math[15].

Псевдокод

 Вход: натуральное число n

Пусть pr - целочисленный массив, поначалу пустой;
      lp - целочисленный массив, индексируемый от 2 до n, заполненный нулями

 для i := 2, 3, 4, ..., до n: 
   если lp[i] = 0:
       lp[i] := i
       pr[] += {i}
   для p из pr пока plp[i] и p*in:
       lp[p*i] := p

Выход: все числа в массиве pr.

Сложность алгоритма на практике

Решето Эратосфена является популярным способом оценки производительности компьютера.[16] Как видно из вышеизложенного доказательства сложности алгоритма, избавившись от константы и слагаемого очень близкого к нулюШаблон:Math, временная сложность вычисления всех простых чисел меньше Шаблон:Mvar аппроксимируется следующим соотношением Шаблон:Math. Однако алгоритм имеет экспоненциальную временную сложность в отношении размера входных данных, что делает его псевдополиномиальным алгоритмом. Памяти же для базового алгоритма требуется Шаблон:Math.[17]

Сегментированная версия имеет ту же операционную сложность Шаблон:Math,[8]. что и несегментированная версия, но уменьшает потребность в используемой памяти до размера сегмента (размер сегмента значительно меньше размера всего массива простых чисел), который равен Шаблон:Math.[18] Также существует очень редко встречающееся на практике оптимизированное сегментированное решето Эратосфена. Оно строится за Шаблон:Math операций и занимает Шаблон:Math бит в памяти.[17][19][18]

На практике оказывается, что оценка Шаблон:Math не очень точна даже для максимальных практических диапазонов таких как 1016.[19] Ознакомившись с вышеописанным доказательством сложности, нетрудно понять откуда взялась неточность оценки. Расхождение с оценкой можно объяснить следующим образом: в пределах данного практического диапазона просеивания существенное влияние оказывают постоянные смещения.[8] Таким образом очень медленно растущий член Шаблон:Math не становится достаточно большим, чтобы константами можно было справедливо пренебречь, до тех пор пока Шаблон:Mvar не приблизится к бесконечности, что естественно выходит за границы любого прикладного диапазона просеивания.[8] Данный факт показывает, что для актуальных на сегодняшний день входных данных производительность решета Эратосфена намного лучше, чем следовало ожидать, используя только асимптотические оценки временной сложности.[19]

Решето Эратосфена работает быстрее, чем часто сравниваемое с ним решето Аткина только для значений Шаблон:Mvar меньших 10 10 .[20] Сказанное справедливо при условии, что операции занимают примерно одно и то же время в циклах центрального процессора, а это является разумным предположением для одного алгоритма, работающего с огромным битовым массивом.[21] С учетом этого предположения получается, что сито Аткина быстрее чем максимально факторизованное решето Эратосфена для Шаблон:Mvar свыше 10 13 , но при таких диапазонах просеивания потребуется занять огромное пространство в оперативной памяти, даже если была использована «битовая упаковка».[21] Однако раздел о сегментированной версии решета Эратосфена показывает, что предположение о сохранении равенства во времени, затрачиваемом на одну операцию, между двумя алгоритмами не выполняется при сегментации.[13][20] В свою очередь это приводит к тому, что решето Аткина (несегментированное) работает медленнее, чем сегментированное решето Эратосфена с увеличением диапазона просеивания, за счёт уменьшения времени на операцию для второго.

Использование нотации Шаблон:Math большого также не является правильным способом сравнения практических характеристик даже для вариаций решета Эратосфена, поскольку данная нотация игнорирует константы и смещения, которые могут быть очень значительными для прикладных диапазонов.[8] Например, одна из вариаций решета Эратосфена известная как решето Притчарда[17] имеет производительность Шаблон:Math,[19] но её базовая реализация требует либо алгоритма «одного большого массива»[18] (то есть использования обычного массива, в котором хранятся все числа до Шаблон:Math), который ограничивает его диапазон использования до объёма доступной памяти, либо он должен быть сегментирован для уменьшения использования памяти. Работа Притчарда уменьшила требования к памяти до предела, но платой за данное улучшение по памяти является усложнение вычислений, что приводит увеличению операционной сложности алгоритмов.[19]

Популярным способом ускорения алгоритмов, работающих с большими массивами чисел, является разного рода факторизация.[22] Применение методов факторизации даёт значительное уменьшение операционной сложности за счёт оптимизации входного массива данных.[23][22] Для индексных алгоритмов часто используют кольцевую факторизацию.[23][24] Рассматриваемые в данной статье алгоритмы нахождения всех простых чисел до заданного Шаблон:Math подобные решету Эратосфена относятся к индексным, что позволяет применять к ним метод кольцевой факторизации.[25]

Несмотря на теоретическое ускорение, получаемое с помощью кольцевой факторизации, на практике существуют факторы, которые не учитываются при расчётах, но способны оказать существенное влияние на поведение алгоритма, что в результате может не дать ожидаемого прироста в быстродействии.[26] Рассмотрим наиболее существенные из них:

  • Шаблон:Math При аналитических расчётах предполагается, что скорость выполнения арифметических операций равноценна. Но на самом деле это не так, и умножение, и деление выполняются гораздо медленнее, чем сложение и вычитание. Таким образом данный фактор никак не влияет на решето Эратосфена, поскольку оно использует только сложение и вычитание, но является весьма существенным для решета Питчарда (один из результатов усложнения вычислений упомянутого выше).[27]
  • Шаблон:Math Компилятор оптимизирует на стадии компиляции все программы для более корректного исполнения машиной. Но часто бывает очень сложно сказать, какой вклад даёт данная оптимизация, и будет ли этот вклад одинаковым для двух различных алгоритмов.[28]
  • Шаблон:Math Процессор использует кэш, чтобы ускорить извлечение инструкций и данных из памяти. Наличие кэша приводит к тому, что программы, использующие локализованные ссылки, будут работать быстрее. Но алгоритмы просеивания простых чисел, которые используют факторизацию высокой степени, часто генерируют случайные ссылки в память, что снижает их производительность.[28]

Для наглядности представления вклада факторизации в производительность алгоритмов просеивания ниже приведены две таблицы. В таблицах приведены результаты измерения реального времени исполнения решета Эратосфена и решета Питчарда в секундах для разных диапазонов Шаблон:Math и разных степеней кольцевой факторизации. Шаблон:Math и Шаблон:Math обозначения для решета Эратосфена и Питчарда соответственно, где индекс Шаблон:Math означает степень кольцевой факторизации. Шаблон:Math и Шаблон:Math означают отсутствие факторизации.[28]

Шаблон:Mvar Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math
500 0.00043 0.00029 0.00027 0.00048 0.00140 0.01035
5000 0.00473 0.00305 0.00254 0.00293 0.00551 0.03207
50000 0.05156 0.03281 0.02617 0.02578 0.03164 0.10663
500000 0.55817 0.35037 0.28240 0.28358 0.28397 0.47028
5000000 6.06719 3.82905 3.20722 3.25214 3.28104 3.38455

Из таблицы видно, что лучшую производительность имеет решето Эратосфена со средней степенью факторизации Шаблон:Math. Данный факт можно объяснить влиянием кэш-фактора, рассмотренного выше, на алгоритмы с высокой степенью факторизации.

Шаблон:Mvar Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math Шаблон:Math
500 0.00147 0.00074 0.00050 0.00051 0.00095 0.00649
5000 0.01519 0.00777 0.00527 0.00453 0.00430 0.00973
50000 0.15507 0.08203 0.05664 0.04843 0.04180 0.04413
500000 1.60732 0.86254 0.61597 0.53825 0.47146 0.43787
5000000 16.47706 9.00177 6.57146 5.83518 5.27427 4.88251

С увеличением Шаблон:Math соотношение времен становится всё больше в пользу решета Эратосфена, а на диапазоне Шаблон:Math = 5000000 оно стабильно быстрее при любых факторизациях. Данный факт ещё раз подтверждает проигрыш в быстродействии решета Питчарда из-за сложных вычислений.[19]

См. также

Примечания

Шаблон:Примечания

Литература

Ссылки

Шаблон:Wikibooks

Шаблон:ВС

  1. 1,0 1,1 Никомах Герасский, Введение в арифметику, I, XIII, 2. по-гречески, по-русски
  2. Шаблон:Книга
  3. 3,0 3,1 3,2 Horsley, Rev. Samuel, F. R. S., "Шаблон:Lang or, The Sieve of Eratosthenes. Being an Account of His Method of Finding All the Prime Numbers, " Philosophical Transactions (1683—1775), Vol. 62. (1772), pp. 327—347 Шаблон:Wayback.
  4. Шаблон:Книга, p. 16.
  5. Jonathan Sorenson, An Introduction to Prime Number Sieves, Computer Sciences Technical Report #909, Department of Computer Sciences University of Wisconsin-Madison, January 2 1990 (the use of optimization of starting from squares, and thus using only the numbers whose square is below the upper limit, is shown).
  6. Pritchard, Paul, "Linear prime-number sieves: a family tree, " Sci. Comput. Programming 9:1 (1987), pp. 17-35.
  7. Строго говоря, внутренний цикл выполняется для каждого простого <math>p \in \mathbb{P}\colon p \le (\sqrt n)</math>, но <math>O(\log (\log n))</math> = <math>O(\log (\log (\sqrt n)))</math>, поэтому, традиционно, для краткости, квадратный корень опускают.
  8. 8,0 8,1 8,2 8,3 8,4 Hardy and Wright "An Introduction to the Theory of Numbers, p. 349
  9. 9,0 9,1 O’Neill, Melissa E., «The Genuine Sieve of Eratosthenes» Шаблон:Wayback, Journal of Functional Programming, Published online by Cambridge University Press 9 October 2008 Шаблон:Doi.
  10. 10,0 10,1 Colin Runciman, «FUNCTIONAL PEARL: Lazy wheel sieves and spirals of primes», Journal of Functional Programming, Volume 7 Issue 2, March 1997; также здесь Шаблон:Wayback.
  11. Turner, David A. SASL language manual. Tech. rept. CS/75/1. Department of Computational Science, University of St. Andrews 1975. (primes = sieve [2..]; sieve (p:nos) = p:sieve (remove (multsof p) nos); remove m = filter (not . m); multsof p n = rem n p==0)
  12. Crandall & Pomerance, Prime Numbers: A Computational Perspective, second edition, Springer: 2005, pp. 121-24.
  13. 13,0 13,1 13,2 Шаблон:Статья
  14. J. Sorenson, The pseudosquares prime sieve Шаблон:Wayback, Proceedings of the 7th International Symposium on Algorithmic Number Theory. (ANTS-VII, 2006).
  15. David Gries, Jayadev Misra. A Linear Sieve Algorithm for Finding Prime Numbers [1978]
  16. Шаблон:Cite news
  17. 17,0 17,1 17,2 Paul Pritchard, A sublinear additive sieve for finding prime numbers, Communications of the ACM 24 (1981), 18-23. Шаблон:MR
  18. 18,0 18,1 18,2 Paul Pritchard, Fast compact prime number sieves (among others), Journal of Algorithms 4 (1983), 332—344. Шаблон:MR
  19. 19,0 19,1 19,2 19,3 19,4 19,5 Paul Pritchard, Explaining the wheel sieve, Acta Informatica 17 (1982), 477—485. Шаблон:MR
  20. 20,0 20,1 Шаблон:Статья
  21. 21,0 21,1 Шаблон:Статья
  22. 22,0 22,1 Шаблон:Статья
  23. 23,0 23,1 Шаблон:Статья
  24. Шаблон:Статья
  25. Шаблон:Статья
  26. Шаблон:Статья
  27. Шаблон:Статья
  28. 28,0 28,1 28,2 Шаблон:Статья