The problem
This is problem 5 from the Project Euler.
2520 is the smallest number that can be divided by each of the numbers from 1 to 10 without any remainder.
What is the smallest positive number that is evenly divisible by all of the numbers from 1 to
n
?
Multiply prime numbers
I know it is something to do with Lowest Common Multiple
and prime numbers
; unfortunately, LCM is something I haven't seen for almost 20 years. So let's recap what LCM is.
I don't know about you, but online resources didn't help me much. I will do the trial and error way to figure out the patterns.
We are given a few test cases:
input(5) = 60
input(7) = 420
input(10) = 2520
I know I have to use prime numbers, so let me multiply some prime numbers to achieve test case results:
input(5) -> 2 * 3 * 5 = 30
input(7) -> 2 * 3 * 5 * 7 = 210
I didn't get the same results as the test case, but the differences are apparent. We need to double 30
and 210
for input(5)
and input(7)
respectively.
Some prime numbers have to be multiplied a few times, and we will get the same results as the test cases:
input(5) = 60
= 1 * 2 * 3 * 2 * 5
= 2^2 * 3 * 5
= 2^2 * 3^1 * 5^1
input(7) = 420
= 1 * 2 * 3 * 2 * 5 * 7
= 2^2 * 3 * 5 * 7
= 2^2 * 3^1 * 5^1 * 7^1
input(10) = 2520
= 1 * 2 * 3 * 2 * 5 * 2 * 7 * 3
= 2^3 * 3^2 * 5 * 7
= 2^3 * 3^2 * 5^1 * 7^1
So the question here is to figure out the number of times each prime number has to be multiplied by. That is:
for each prime number:
product *= Math.pow(prime, ?)
Since I have to work with multiplication and exponent, I am might need logarithm in my algorithm too.
The idea here is to find the maximum exponent for each prime number, that is to find out the number of times each prime number has to be multiplied by dividing logarithm of the input number n
by logarithm of the prime number. Since all exponent is whole numbers, we use Math.floor
to round down the exponent.
Math.floor [ Math.log(number) / Math.log(prime) ]
So, for-loop over all primes numbers prime_numbers
less than given number n
and multiply them together with a reduce
function:
var multiply_them = prime_numbers.reduce(function(product, prime) {
return product * Math.pow(prime, Math.floor(Math.log(n) / Math.log(prime)));
}, 1);
Get all prime numbers less than given number
This is pretty straight forward.
The number 2
is the only even prime number. We can skip all even numbers, as all other even numbers are not prime numbers.
If a number num
divides every other prime number less than num
does not produce any decimal, then num
is a prime number.
Return list_of_prime_nums
containing every prime number smaller than n
.
// generate and return prime numbers less than `n`
// skipping all even numbers because `2` is the only even prime number
function generate_prime_from_less_than_n(n){
var list_of_prime_nums = [2];
for(var i=3; i<=n; i+=2){
var is_prime = true;
for(var j=0; j<list_of_prime_nums.length; j++){
var check_is_prime = i % list_of_prime_nums[j];
if(check_is_prime === 0){
is_prime = false;
break;
}
}
if(is_prime){
list_of_prime_nums.push(i);
}
}
return list_of_prime_nums;
}
Combine everything
// list of numbers we wanna test
var test_values = [5, 7, 10, 13, 20];
// this function execute the code and records the time to execute
function run_function(func, test_values) {
for(var i in test_values){
console.log('Test value:', test_values[i]);
var t0 = performance.now();
console.log('Output:', func(test_values[i]));
var t1 = performance.now();
console.log("Took " + (t1 - t0) + " ms");
console.log();
}
}
function smallestMult(n) {
var prime_numbers = generate_prime_from_less_than_n(n);
console.log('prime numbers:', prime_numbers);
var multiply_them = prime_numbers.reduce(function(product, prime) {
return product * Math.pow(prime, Math.floor(Math.log(n) / Math.log(prime)));
}, 1);
return multiply_them;
}
// generate and return prime numbers less than `n`
// skipping all even numbers because `2` is the only even prime number
function generate_prime_from_less_than_n(n){
var list_of_prime_nums = [2];
for(var i=3; i<=n; i+=2){
var is_prime = true;
for(var j=0; j<list_of_prime_nums.length; j++){
var check_is_prime = i % list_of_prime_nums[j];
if(check_is_prime === 0){
is_prime = false;
break;
}
}
if(is_prime){
list_of_prime_nums.push(i);
}
}
return list_of_prime_nums;
}
run_function(smallestMult, test_values);
Output:
Test value: 5
prime numbers: [ 2, 3, 5 ]
Output: 60
Took 0.6450000073527917 ms
Test value: 7
prime numbers: [ 2, 3, 5, 7 ]
Output: 420
Took 0.26000000070780516 ms
Test value: 10
prime numbers: [ 2, 3, 5, 7 ]
Output: 2520
Took 0.16999999934341758 ms
Test value: 13
prime numbers: [ 2, 3, 5, 7, 11, 13 ]
Output: 360360
Took 0.13500000932253897 ms
Test value: 20
prime numbers: [ 2, 3, 5, 7, 11, 13, 17, 19 ]
Output: 232792560
Took 0.13000000035390258 ms
Here's my solution, can it be any better?
This is my Project Euler Challenge journey; anyone wants to do this together? It will be fun and we can learn a thing or two by solving this problem in different ways.
Top comments (0)