June 28, 2021
Suppose you are given a very large number, for example , and you wish to know how many numbers equal to or less than are prime. This is the prime counting function, normally denotes as . In this article we will explore some of the methods to efficiently calculate , and we will benchmark each in Rust.
Checking every number up to for primality
Let's write a naive implementation that we will use as a baseline. Of course, the first approach would be to test the primality of each number from to . First, let's write a naive implementation for is_prime
:
fn is_prime(x: usize) -> bool {
for d in 2..x {
if x % d == 0 {
return false;
}
}
return true;
}
Now, counting primes is as simple as iterating over the numbers that can be prime and testing each one:
pub fn count(n: usize) -> usize {
let mut result = 0;
for x in 2..(n + 1) {
if is_prime(x) {
result += 1;
}
}
return result;
}
As you might expect, this function does terribly:
Begun testing the naive implementation with n = 10^5.
The amount of prime numbers that are less than or equal to 10^5 is 9592.
Elapsed: 1.03s
Finished testing the naive implementation with n = 10^5.
I couldn't even get it to finish calculating in under a minute.
Using the Sieve of Eratosthenes
A better option than the naive implementation is the Sieve of Eratosthenes. There is a nice graphic on its Wikipedia entry explaining how it works, so I won't go into detail. First of all, note that it is clearly faster than checking each prime as the operations are simpler (just addition and multiplication instead of the modulus operator). The problem with this algorithm is that you would need a flag for each number to indicate whether it is prime or not. For large enough this is untenable, as you will soon run out of memory, or the memory will take too long to allocate. Still, you could use the Sieve of Eratosthenes to rapidly compute the primes up to a given limit.
A naive implementation of the sieve in Rust looks like:
pub fn count(n: usize) -> usize {
let mut result = 0;
let mut flags = Vec::with_capacity(n);
for _ in 0..n {
flags.push(true);
}
flags[0] = false;
for i in 0..n {
if flags[i] {
let x = i + 1;
for m in ((i + x)..n).step_by(x) {
flags[m] = false;
}
result += 1;
}
}
return result;
}
A benchmark using proves this function to be much faster than the naive implementation:
Begun testing the sieve implementation with n = 10^6.
The amount of prime numbers that are less than or equal to 10^6 is 78498.
Elapsed: 6.86ms
Finished testing the sieve implementation with n = 10^6.
Sadly, it starts to falters at around :
Begun testing the sieve implementation with n = 10^9.
The amount of prime numbers that are less than or equal to 10^9 is 50847534.
Elapsed: 16.53s
Finished testing the sieve implementation with n = 10^9.
I couldn't get it to run for in under a minute.
This does not imply that the sieve is useless for our purposes of calculating . In fact, we will be using it in the next few sections.
Without knowing all the primes (Legendre's Formula)
To reduce the amount of memory required for our computation we will use Legendre's Formula. It essentially allows you to count primes up to , while only knowing the primes up to . Now let's proceed to prove it.
Let be the amount of numbers less than or equal to , which are not divisible by the first primes. Then we can calculate using the inclusion-exclusion principle:
where all the .
Now, let's prove the following fact: suppose . Then either is , it has a divisor or it is prime. If , we are done. If then it has a divisor , where . Now let . We will now proceed by contradiction. Suppose is the smallest divisor of larger than . Assume said divisor is larger than . Clearly is also a divisor of and thus . But , which is impossible. Therefore must not exceed .
From the previous fact we have that:
Which can be rearranged to find . This is Legendre's Formula.
Now let's write an implementation for Legendre's Formula. First we need to find all primes up to . This can be done in multiple ways, but the sieve of Eratosthenes is particularly effective and simple:
fn isqrt(x: usize) -> usize {
return (x as f64).sqrt() as usize;
}
// Calculate primes up to isqrt(n)
fn get_primes(n: usize) -> Vec<usize> {
let l = isqrt(n);
let capacity = (1.5 * (l as f64) / (l as f64).ln()) as usize;
let mut primes = Vec::with_capacity(capacity);
let mut flags = Vec::with_capacity(l);
for _ in 0..l {
flags.push(true);
}
flags[0] = false;
for i in 0..l {
if flags[i] {
let x = i + 1;
for m in ((i + x)..l).step_by(x) {
flags[m] = false;
}
primes.push(x);
}
}
return primes;
}
Notice that we preallocate as the capacity for the primes
vector. This is because is upper bounded by this quantity, as shown on Wikipedia.
And now we need to calculate the sum. Because Legendre's Formula is an infinite sum, we need to find a stopping point. We can get a clear cutoff point by noting that, in each sum, the smallest divisor is the product of the first primes. At some point this product is going to get larger than , at which point all terms in the sum will be . This logic is implemented with the function get_max_depth
:
fn get_max_depth(n: usize, primes: &Vec<usize>) -> usize {
let mut max_depth = 0;
let mut min_product = 1;
while max_depth < primes.len() {
min_product *= primes[max_depth];
if min_product > n { break; }
max_depth += 1;
}
return max_depth;
}
Finally, we need to implement the function that calculates each of the sums. Because the number of primes that you must use in each sum is not fixed, we need to implement this function using recursion. I don't want to go into details of how to do that, so this is what the function ends up looking like:
fn calculate_sum(n: usize, primes: &Vec<usize>, depth: usize, level: usize, maybe_last_index: Option<usize>, product: usize) -> usize {
if depth < level { return n / product; }
let mut result = 0;
let start_index = match maybe_last_index {
Some(last_index) => last_index + 1,
None => 0,
};
let end_index = primes.len() - depth + level;
for index in start_index..end_index {
let next_level = level + 1;
let next_last_index = Some(index);
let next_product = product * primes[index];
if next_product > n { break; }
result += calculate_sum(n, primes, depth, next_level, next_last_index, next_product);
}
return result;
}
Finally let's write the count
function, which is now trivial to implement:
pub fn count(n: usize) -> usize {
let mut result: isize = 0;
let primes = get_primes(n);
let max_depth = get_max_depth(n, &primes);
// Use Legendre's Formula
result += primes.len() as isize;
result -= 1;
for depth in 0..(max_depth + 1) {
let term = calculate_sum(n, &primes, depth, 1, None, 1) as isize;
if depth % 2 == 0 {
result += term;
} else {
result -= term;
}
}
return result as usize;
}
Benchmarks show us that Legendre's formula is faster than the Sieve of Eratosthenes for :
Begun testing the legendre implementation with n = 10^9.
The amount of prime numbers that are less than or equal to 10^9 is 50847534.
Elapsed: 4.29s
Finished testing the legendre implementation with n = 10^9.
Sadly, Legendre's formula scales poorly. With we almost reached the minute mark:
Begun testing the legendre implementation with n = 10^10.
The amount of prime numbers that are less than or equal to 10^10 is 455052511.
Elapsed: 55.34s
Finished testing the legendre implementation with n = 10^10.
so it does not make sense to attempt .
This failure to scale is due to the fact Legendre's formula has to iterate over millions of choices of prime multiplications. Thankfully, our next approach will get rid of this problem.
Meissel–Lehmer's algorithm
Now comes the interesting part of the post. In this article D. H. Lehmer introduces a very efficient formula for computing . Let's prove it.
Let be the amount of numbers less than or equal to , which are not divisible by any of the first primes and which have exactly prime factors. Then:
for some finite , as at some point the product of enough primes must be larger than .
Note that:
Joining and and using the fact that we get:
And with the right choice of , and an efficient method to calculate and the , we have an efficient formula for .
Let's start by computing . Let be the -th prime. Then:
Now let and . Then we can rewrite the above as follows:
Now for , a similar procedure yields:
where .
Now let . Choose or more primes larger than . Then the product of those primes is larger than . Therefore . Thus, we can derive several distinct formulas through the choice of .
If we set , we get , which is Legendre's formula.
If we set we get or equivalently from plugging :
which is Meissel's formula.
But the choice of we are interested in is . With this choice the formula expands to also include as follows:
which is Meissel-Lehmer's formula. Essentially, it is a recursive version of the prime counting function, and therefore we can speed up the computation by knowing the values of up to a large enough using the other algorithms. Now let's find a way to calculate efficiently. First notice the following recursion:
To prove that this is true, note that the amount of numbers that are divisible by but not by is given by . On the other hand, this amount can be expressed as:
which proves the recursion. Before using the recursion, let's prove a few base cases:
Case 1: . This follows directly from the definition.
Case 2: . These are the numbers that are not divisble by .
Case 3: . This follows from the inclusion-exclusion principle using the primes and .
Case 4: If , then . Clearly, all are divisible by one of , except for the number .
With the base cases out of the way let's write an expansion for using the recursion formula:
And with that we have all the parts we need to begin implementing Meissel-Lehmer formula.
To begin implementing, first we need a list of all primes up to , and a way to calculate , for "small" values of . Finding the primes can be done using the Sieve of Eratosthenes, and once we have the primes we can perform binary search on the list of primes to calculate up to the limit we used to calculate the primes. Our implementation looks like this:
struct PrimeTable {
limit: usize,
primes: Vec<usize>,
}
impl PrimeTable {
pub fn new(limit: usize) -> PrimeTable {
let capacity = (1.5 * (limit as f64) / (limit as f64).ln()) as usize;
let mut primes = Vec::with_capacity(capacity);
let mut flags = Vec::with_capacity(limit);
for _ in 0..limit {
flags.push(true);
}
flags[0] = false;
for i in 0..limit {
if flags[i] {
let x = i + 1;
for m in ((i + x)..limit).step_by(x) {
flags[m] = false;
}
primes.push(x);
}
}
return PrimeTable { limit, primes };
}
pub fn get_prime_count(&self, n: usize) -> Option<usize> {
if n > self.limit { return None; }
let mut start = 0;
let mut end = self.primes.len() - 1;
while end - start > 1 {
let middle = (start + end) / 2;
if self.primes[middle] <= n {
start = middle;
} else {
end = middle;
}
}
if start == end || self.primes[end] > n {
return Some(start + 1);
} else {
return Some(end + 1);
}
}
pub fn get_prime(&self, i: usize) -> usize {
return self.primes[i - 1];
}
}
Notice that this table of primes can be reused in the implementation of the recursive version of , as that function requires a list of primes. Our implementation of follows trivially from and the base cases:
fn phi(n: usize, a: usize, table: &PrimeTable) -> usize {
if a == 0 {
return n;
} else if a == 1 {
return n - (n / 2);
} else if n <= table.get_prime(a) {
return 1;
}
let mut result = n - (n / 2) - (n / 3) + (n / 6);
for i in 3..(a + 1) {
result -= phi(n / table.get_prime(i), i - 1, &table);
}
return result;
}
We need a way to calculate the integer -th root of a number. For our purposes, we used the following implementation:
fn integer_nth_root(n: usize, r: f64) -> usize {
return (n as f64).powf(1.0 / r) as usize;
}
Because f64
is not an "exact" type, this implementation may fail for some inputs. Nevertheless, it is good enough for the scale at which we are working.
Finally, Meissel-Lehmer's implementation follows trivially from :
fn meissel_lehmer(n: usize, table: &PrimeTable) -> usize {
if let Some(result) = table.get_prime_count(n) {
return result;
}
let a = meissel_lehmer(integer_nth_root(n, 4.0), &table);
let b = meissel_lehmer(integer_nth_root(n, 2.0), &table);
let c = meissel_lehmer(integer_nth_root(n, 3.0), &table);
let mut result = phi(n, a, &table) + ((b + a - 2) * (b - a + 1)) / 2;
// Calculate P_2
for i in (a + 1)..(b + 1) {
result -= meissel_lehmer(n / table.get_prime(i), &table);
}
// Calculate P_3
for i in (a + 1)..(c + 1) {
let b_i = meissel_lehmer(integer_nth_root(n / table.get_prime(i), 2.0), &table);
for j in i..(b_i + 1) {
let denominator = table.get_prime(i) * table.get_prime(j);
result -= meissel_lehmer(n / denominator, &table) - (j - 1);
}
}
return result;
}
With all of these parts, our count
function only needs to initialize a PrimeTable
, and use meissel_lehmer
on the given input.
pub fn count(n: usize) -> usize {
let limit = std::cmp::min(n, usize::pow(10, 9));
let prime_table = PrimeTable::new(limit);
return meissel_lehmer(n, &prime_table);
}
Because our initial problem is calculating , we chose to calculate the prime list up to , even though we only needed to do so up to . This reduces the number of meissel_lehmer
calls, which helps speed up the computation.
Because our implementation essentially uses the sieve algorithm to calculate primes up to , we know it will take at least ~15s to finish. Let's benchmark :
Begun testing the meissel_lehmer implementation with n = 10^10.
The amount of prime numbers that are less than or equal to 10^10 is 455052511.
Elapsed: 16.09s
Finished testing the meissel_lehmer implementation with n = 10^10.
Notice that it does not take much to finish after the 15s mark. Let's see if this extends all the way up to :
Begun testing the meissel_lehmer implementation with n = 10^12.
The amount of prime numbers that are less than or equal to 10^12 is 37607912018.
Elapsed: 18.97s
Finished testing the meissel_lehmer implementation with n = 10^12.
And it does!
Conclusion
In this post we've proven and implemented an efficient method for calculating for large enough . Hopefully the reader has learned a new thing or two in this post!
If you want to check out the code we used, head over to my Github.