Open In App

Sieve of Atkin

Improve
Improve
Like Article
Like
Save
Share
Report

Given a limit, print all prime numbers smaller than or equal to the given limit.

Examples: 

Input:  limit = 10
Output: 2, 3, 5, 7

Input:  limit = 20
Output: 2, 3, 5, 7, 11, 13, 17, 19 

Other Approaches: The prime numbers till a limit can be generated using other algorithms, such as: 

Sieve of Atkin Approach: The sieve of Atkin is a modern algorithm for finding all prime numbers up to a specified integer. 

Sieve of Atkin vs Sieve of Eratosthenes: 

Compared with the ancient Sieve of Eratosthenes, which marks off multiples of primes, it does some preliminary work and then marks off multiples of squares of primes, that’s why it has a better theoretical asymptotic complexity with Complexity of (N / (log (log N)))

How Sieve of Atkin algorithm works: 

The Sieve of Atkin algorithm works similarly to Sieve of Eratosthenes to filter out the composite numbers from a list of numbers, but this algorithm works in terms of modulo-60 remainders

So we first assume all the numbers within limit to be composite, and then apply filter or sieve on them. If while any filter, the number appears to be prime, we mark it as prime and move on to the next number.

The filter or sieve in this algorithms works mainly 4 cases or layers:

  • Case 1: If limit is greater than 2 or 3:
    • The algorithm treats 2, and 3 as special cases and just adds them to the set of primes to start with.
  • Case 2: if 4x2+y2=n is odd and modulo-12 remainder is 1 or 5
    • Since all numbers with modulo-60 remainders 1, 13, 17, 29, 37, 41, 49, or 53 have a modulo-12 remainder of 1 or 5. Therefore, for this filter as well, we have to check if the number is 1 or 5 when taken modulo with 12.
    • Also, These numbers are prime if and only if the number of solutions to 4x2+y2=n is odd and the number is square-free. 
    • A square-free integer is one that is not divisible by any perfect square other than 1.
  • Case 3: if 3x2+y2=n is odd and modulo-6 remainder is 1
    • All numbers with modulo-60 remainder 7, 19, 31, or 43 have a modulo-6 remainder of 1. 
    • These numbers are prime if and only if the number of solutions to 3x2 + y2 = n is odd and the number is square-free.
  • Case 4: if 3x2-y2=n is odd and modulo-12 remainder is 11
    • All numbers with modulo-60 remainder 11, 23, 47, or 59 have a modulo-12 remainder of 11. 
    • These numbers are prime if and only if the number of solutions to 3x2 – y2 = n is odd and the number is square-free.
  • Case 5: Filtering out all the residual primes which have not yet been found
    • Due to the filtering of the Sieve of Atkin algorithm, there might be some prime numbers who have been discarded or not found in the above cases.
    • So to find out those, select all non-primes within limit, and mark all their squares as non-primes.

At the end of all of the filters above, the positions in the Sieve with a true value will be the list of primes within limit.

Illustration of Sieve of Atkin algorithm:

Consider limit as 20 and lets see how Sieve of Atkin algorithm generates primes up to 20:

Step 0: The status for all the numbers at the start is false. The special numbers are 2, 3, and 5, which are known to be prime.

Step 1: Generate values for the conditions.  

atkins

Step 2: Flipping the status according to condition.
The above values of n in the table generated in the x, y loop will be tested for modulo conditions.

  • Column 1: if (column1 value) % 12 == 1 or 5, then flip the sieve status for that number.
  • Column 2: if (column2 value) % 12 == 7, then flip the sieve status for that number.
  • Column 3: if (column3 value) % 12 == 11, then flip the sieve status for that number.

Note: Notice that we are taking mod with 12 in place of 60. This is because if we take mod 60 then we have to consider as many r as 1, 13, 17, 29, 37, 41, 49, or 53 and for all these r, mod 12 is 1 or 5. (done only to reduce the expression size)

Step 3: Checking for Square free Condition: 
If any number in our list is the square of any number, then remove it.

Step 4: Creating an array of prime numbers for which status is true. 
i.e. 2 3 5 7 11 13 17 19

Step 5: Print the output on the screen.

Sieve of Atkin algorithm step-by-step:

  1. Create a results list, filled with 2, 3, and 5.
  2. Create a sieve list with an entry for each positive integer; all entries in this list should initially be marked non-prime.
  3. For each entry number n in the sieve list, with modulo-sixty remainder r: 
    1. If r is 1, 13, 17, 29, 37, 41, 49, or 53, flip the entry for each possible solution to 4x2 + y2 = n.
    2. If r is 7, 19, 31, or 43, flip the entry for each possible solution to 3x2 + y2 = n.
    3. If r is 11, 23, 47, or 59, flip the entry for each possible solution to 3x2 – y2 = n when x > y.
    4. If r is something else, ignore it completely…
  4. Start with the lowest number in the sieve list.
  5. Take the next number in the sieve list, still marked prime.
  6. Include the number in the results list.
  7. Square the number and mark all multiples of that square as non-prime. Note that the multiples that can be factored by 2, 3, or 5 need not be marked, as these will be ignored in the final enumeration of primes.
  8. Repeat steps four through seven.

Below is the implementation of the above algorithm. 

C++




// C++ program for implementation
// of Sieve of Atkin
#include <bits/stdc++.h>
using namespace std;
 
// Function to generate primes
// till limit using Sieve of Atkin
void SieveOfAtkin(int limit)
{
    // Initialise the sieve array
    // with initial false values
    bool sieve[limit + 1];
    for (int i = 0; i <= limit; i++)
        sieve[i] = false;
 
    // 2 and 3 are known to be prime
    if (limit > 2)
        sieve[2] = true;
    if (limit > 3)
        sieve[3] = true;
   
    /* Mark sieve[n] is true if one
       of the following is true:
    a) n = (4*x*x)+(y*y) has odd number of
       solutions, i.e., there exist
       odd number of distinct pairs (x, y)
       that satisfy the equation and
        n % 12 = 1 or n % 12 = 5.
    b) n = (3*x*x)+(y*y) has odd number of
       solutions and n % 12 = 7
    c) n = (3*x*x)-(y*y) has odd number of
       solutions, x > y and n % 12 = 11 */
    for (int x = 1; x * x <= limit; x++) {
        for (int y = 1; y * y <= limit; y++) {
 
            // Condition 1
            int n = (4 * x * x) + (y * y);
            if (n <= limit
                && (n % 12 == 1 || n % 12 == 5))
                sieve[n] ^= true;
 
            // Condition 2
            n = (3 * x * x) + (y * y);
            if (n <= limit && n % 12 == 7)
                sieve[n] ^= true;
 
            // Condition 3
            n = (3 * x * x) - (y * y);
            if (x > y && n <= limit
                && n % 12 == 11)
                sieve[n] ^= true;
        }
    }
 
    // Mark all multiples
    // of squares as non-prime
    for (int r = 5; r * r <= limit; r++) {
        if (sieve[r]) {
            for (int i = r * r; i <= limit; i += r * r)
                sieve[i] = false;
        }
    }
 
    // Print primes using sieve[]
    for (int a = 1; a <= limit; a++)
        if (sieve[a])
            cout << a << " ";
    cout << "\n";
}
 
// Driver program
int main(void)
{
    int limit = 19;
    SieveOfAtkin(limit);
    return 0;
}


Java




// Java program for implementation
// of Sieve of Atkin
class GFG {
 
    static void SieveOfAtkin(int limit)
    {
        // 2 and 3 are known to be prime
        if (limit > 2)
            System.out.print(2 + " ");
 
        if (limit > 3)
            System.out.print(3 + " ");
 
        // Initialise the sieve array with
        // false values
        boolean sieve[] = new boolean[limit+1];
 
        for (int i = 0; i <= limit; i++)
            sieve[i] = false;
 
        /* Mark sieve[n] is true if one of the
        following is true:
        a) n = (4*x*x)+(y*y) has odd number
           of solutions, i.e., there exist
           odd number of distinct pairs
           (x, y) that satisfy the equation
           and    n % 12 = 1 or n % 12 = 5.
        b) n = (3*x*x)+(y*y) has odd number
           of solutions and n % 12 = 7
        c) n = (3*x*x)-(y*y) has odd number
           of solutions, x > y and n % 12 = 11 */
        for (int x = 1; x * x <= limit; x++) {
            for (int y = 1; y * y <= limit; y++) {
 
                // Main part of Sieve of Atkin
                int n = (4 * x * x) + (y * y);
                if (n <= limit
                    && (n % 12 == 1 || n % 12 == 5))
 
                    sieve[n] ^= true;
 
                n = (3 * x * x) + (y * y);
                if (n <= limit && n % 12 == 7)
                    sieve[n] ^= true;
 
                n = (3 * x * x) - (y * y);
                if (x > y && n <= limit
                    && n % 12 == 11)
                    sieve[n] ^= true;
            }
        }
 
        // Mark all multiples of squares as
        // non-prime
        for (int r = 5; r * r <= limit; r++) {
            if (sieve[r]) {
                for (int i = r * r; i <= limit;
                     i += r * r)
                    sieve[i] = false;
            }
        }
 
        // Print primes using sieve[]
        for (int a = 5; a <= limit; a++)
            if (sieve[a])
                System.out.print(a + " ");
        System.out.println();
    }
 
    // Driver code
    public static void main(String[] args)
    {
        int limit = 20;
        SieveOfAtkin(limit);
    }
}
 
// This code is contributed by Anant Agarwal.


Python 3




# Python 3 program for
# implementation of
# Sieve of Atkin
 
def SieveOfAtkin(limit):
    # 2 and 3 are known
    # to be prime
    if limit > 2:
        print(2, end=" ")
    if limit > 3:
        print(3, end=" ")
 
    # Initialise the sieve
    # array with False values
    sieve = [False] * (limit + 1)
    for i in range(0, limit + 1):
        sieve[i] = False
 
    '''Mark sieve[n] is True if
    one of the following is True:
    a) n = (4*x*x)+(y*y) has odd
    number of solutions, i.e.,
    there exist odd number of
    distinct pairs (x, y) that
    satisfy the equation and
    n % 12 = 1 or n % 12 = 5.
    b) n = (3*x*x)+(y*y) has
    odd number of solutions
    and n % 12 = 7
    c) n = (3*x*x)-(y*y) has
    odd number of solutions,
    x > y and n % 12 = 11 '''
    x = 1
    while x * x <= limit:
        y = 1
        while y * y <= limit:
 
            # Main part of
            # Sieve of Atkin
            n = (4 * x * x) + (y * y)
            if (n <= limit and (n % 12 == 1 or
                                n % 12 == 5)):
                sieve[n] ^= True
 
            n = (3 * x * x) + (y * y)
            if n <= limit and n % 12 == 7:
                sieve[n] ^= True
 
            n = (3 * x * x) - (y * y)
            if (x > y and n <= limit and
                    n % 12 == 11):
                sieve[n] ^= True
            y += 1
        x += 1
 
    # Mark all multiples of
    # squares as non-prime
    r = 5
    while r * r <= limit:
        if sieve[r]:
            for i in range(r * r, limit+1, r * r):
                sieve[i] = False
 
        r += 1
 
        # Print primes
    # using sieve[]
    for a in range(5, limit+1):
        if sieve[a]:
            print(a, end=" ")
             
# Driver Code
limit = 20
SieveOfAtkin(limit)
 
# This code is contributed
# by Smitha


C#




// C# program for implementation of Sieve
// of Atkin
using System;
 
class GFG {
 
    static void SieveOfAtkin(int limit)
    {
        // 2 and 3 are known to be prime
        if (limit > 2)
            Console.Write(2 + " ");
 
        if (limit > 3)
            Console.Write(3 + " ");
 
        // Initialise the sieve array with
        // false values
        bool[] sieve = new bool[limit + 1];
 
        for (int i = 0; i <= limit; i++)
            sieve[i] = false;
 
        /* Mark sieve[n] is true if one of the
        following is true:
        a) n = (4*x*x)+(y*y) has odd number
           of solutions, i.e., there exist
           odd number of distinct pairs
           (x, y) that satisfy the equation
           and    n % 12 = 1 or n % 12 = 5.
        b) n = (3*x*x)+(y*y) has odd number
           of solutions and n % 12 = 7
        c) n = (3*x*x)-(y*y) has odd number
           of solutions, x > y and n % 12 = 11 */
        for (int x = 1; x * x <= limit; x++) {
            for (int y = 1; y * y <= limit; y++) {
 
                // Main part of Sieve of Atkin
                int n = (4 * x * x) + (y * y);
                if (n <= limit
                    && (n % 12 == 1 || n % 12 == 5))
 
                    sieve[n] ^= true;
 
                n = (3 * x * x) + (y * y);
                if (n <= limit && n % 12 == 7)
                    sieve[n] ^= true;
 
                n = (3 * x * x) - (y * y);
                if (x > y && n <= limit
                    && n % 12 == 11)
                    sieve[n] ^= true;
            }
        }
 
        // Mark all multiples of squares as
        // non-prime
        for (int r = 5; r * r < limit; r++) {
            if (sieve[r]) {
                for (int i = r * r; i < limit;
                     i += r * r)
                    sieve[i] = false;
            }
        }
 
        // Print primes using sieve[]
        for (int a = 5; a <= limit; a++)
            if (sieve[a])
                Console.Write(a + " ");
        Console.WriteLine();
    }
 
    // Driver code
    public static void Main()
    {
        int limit = 20;
        SieveOfAtkin(limit);
    }
}
 
// This code is contributed by nitin mittal


PHP




<?php
// PHP program for implementation
// of Sieve of Atkin
 
function SieveOfAtkin($limit)
{
     
    // 2 and 3 are known
    // to be prime
    if ($limit > 2)
        echo 2 , " ";
    if ($limit > 3)
        echo 3 , " ";
 
    // Initialise the sieve array
    // with false values
    $sieve[$limit+1] = 0;
    for ($i = 0; $i <= $limit; $i++)
        $sieve[$i] = false;
 
    /* Mark sieve[n] is true if one
       of the following is true:
    a) n = (4*x*x)+(y*y) has odd number of
       solutions, i.e., there exist
       odd number of distinct pairs (x, y)
       that satisfy the equation and
       n % 12 = 1 or n % 12 = 5.
    b) n = (3*x*x)+(y*y) has odd number of
       solutions and n % 12 = 7
    c) n = (3*x*x)-(y*y) has odd number of
       solutions, x > y and n % 12 = 11 */
    for ($x = 1; $x * $x <= $limit; $x++)
    {
        for ($y = 1; $y * $y <= $limit; $y++)
        {
             
            // Main part of Sieve of Atkin
            $n = (4 * $x * $x) + ($y * $y);
            if ($n <= $limit
                && ($n % 12 == 1 || $n % 12 == 5))
                $sieve[$n] ^= true;
 
            $n = (3 * $x * $x) + ($y * $y);
            if ($n <= $limit && $n % 12 == 7)
                $sieve[$n] = true;
 
            $n = (3 * $x * $x) - ($y * $y);
            if ($x > $y && $n <= $limit
                && $n % 12 == 11)
                $sieve[$n] ^= true;
        }
    }
 
    // Mark all multiples of
    // squares as non-prime
    for ($r = 5; $r * $r <= $limit; $r++) {
        if ($sieve[$r]) {
            for ($i = $r * $r; $i <= $limit;
                             $i += $r * $r)
                $sieve[$i] = false;
        }
    }
 
    // Print primes
    // using sieve[]
    for ($a = 5; $a <= $limit; $a++)
        if ($sieve[$a])
            echo $a , " ";
    echo "\n";
}
 
    // Driver Code
    $limit = 20;
    SieveOfAtkin($limit);
 
// This code is contributed by nitin mittal.
?>


Javascript




<script>
// Javascript program for implementation
// of Sieve of Atkin
 
function SieveOfAtkin(limit)
{
     
    // 2 and 3 are known
    // to be prime
    if (limit > 2)
        document.write(2 + " ");
    if (limit > 3)
        document.write(3 + " ");
 
    // Initialise the sieve array
    // with false values
    let sieve = new Array()
    sieve[limit+1] = 0;
    for (let i = 0; i <= limit; i++)
        sieve[i] = false;
 
    /* Mark sieve[n] is true if one
       of the following is true:
    a) n = (4*x*x)+(y*y) has odd number of
       solutions, i.e., there exist
       odd number of distinct pairs (x, y)
       that satisfy the equation and
       n % 12 = 1 or n % 12 = 5.
    b) n = (3*x*x)+(y*y) has odd number of
       solutions and n % 12 = 7
    c) n = (3*x*x)-(y*y) has odd number of
       solutions, x > y and n % 12 = 11
    */
    for (let x = 1; x * x <= limit; x++)
    {
        for (let y = 1; y * y <= limit; y++)
        {
             
            // Main part of Sieve of Atkin
            let n = (4 * x * x) + (y * y);
            if (n <= limit && (n % 12 == 1 ||
                                n % 12 == 5))
                sieve[n] ^= true;
 
            n = (3 * x * x) + (y * y);
            if (n <= limit && n % 12 == 7)
                sieve[n] ^= true;
 
            n = (3 * x * x) - (y * y);
            if (x > y && n <= limit &&
                            n % 12 == 11)
                sieve[n] ^= true;
        }
    }
 
    // Mark all multiples of
    // squares as non-prime
    for (let r = 5; r * r <= limit; r++) {
        if (sieve[r]) {
            for (i = r * r; i <= limit;
                            i += r * r)
                sieve[i] = false;
        }
    }
 
    // Print primes
    // using sieve[]
    for (let a = 5; a <= limit; a++)
        if (sieve[a])
            document.write(a , " ");
        document.write("<br>");
}
 
    // Driver Code
    let limit = 20;
    SieveOfAtkin(limit);
 
// This code is contributed by nitin gfgking
 
</script>


Output

2 3 5 7 11 13 17 19 

Time complexity: O(limit)
Auxiliary space: O(limit)

 



Last Updated : 31 Mar, 2023
Like Article
Save Article
Previous
Next
Share your thoughts in the comments
Similar Reads