Ottimizzazione della Velocità tramite Memoria in JavaScript
Ci sono situazioni in cui è possibile sacrificare la memoria RAM per aumentare le prestazioni.
Vediamo un esempio. Il codice seguente trova i numeri amicabili in un intervallo prefissato:
console.log(getFriendly(9000));
function getFriendly(range) {
let res = [];
for (let i = 1; i <= range; i++) {
for (let j = 1; j < range; j++) {
if (isFriendly(i, j)) {
res.push([i, j]);
}
}
}
return res;
}
function isFriendly(num1, num2) {
let sum1 = getSum(getOwnDivisors(num1));
let sum2 = getSum(getOwnDivisors(num2));
return sum1 == num2 && sum2 == num1;
}
function getOwnDivisors(num) {
let res = [];
for (let i = 1; i < num; i++) {
if (num % i === 0) {
res.push(i);
}
}
return res;
}
function getSum(arr) {
let sum = 0;
for (let elem of arr) {
sum += elem;
}
return sum;
}
Il codice sopra non è ottimale.
Esegue un numero elevato di operazioni
e con un intervallo fino a 9000
la pagina del browser si bloccherà.
Il problema di questo codice è che
per ogni numero calcoliamo la somma dei suoi
divisori moltissime volte, tante
quanti sono i numeri totali che controlliamo.
Ciò significa che nel nostro caso
per qualsiasi numero la somma dei suoi divisori
sarà calcolata 9000 volte.
Non c'è da stupirsi che tutto si blocchi.
Ottimizziamo. Per cominciare creiamo una funzione che calcoli direttamente la somma dei divisori, senza salvarli in un array:
function getOwnDivisorsSum(num) {
let sum = 0;
for (let i = 1; i < num; i++) {
if (num % i === 0) {
sum += i;
}
}
return sum;
}
Ora è il momento di sacrificare la memoria RAM. Creiamo una funzione che calcoli in anticipo una sola volta la somma dei divisori di tutti i numeri nell'intervallo specificato e li salvi in un array.
Il risultato della nostra funzione sarà un array, in cui la chiave sarà il numero (meno uno), e il valore sarà la somma dei suoi divisori. Implementiamo la nostra funzione:
function getAllSum(range) {
let arr = [];
for (let i = 1; i <= range; i++) {
arr.push(getOwnDivisorsSum(i));
}
return arr;
}
Ora per verificare l'amicabilità non calcoleremo ogni volta la somma dei divisori dei numeri, ma semplicemente prenderemo quella già calcolata dall'array:
function getFriendly(range) {
let sums = getAllSum(range); // [1, 2, 6...]
let res = [];
for (let i = 0; i < sums.length; i++) {
for (let j = i; j < sums.length; j++) {
let sum1 = sums[i];
let sum2 = sums[j];
let num1 = i + 1;
let num2 = j + 1;
if (num1 == sum2 && num2 == sum1) {
res.push([num1, num2]);
}
}
}
return res;
}
Mettiamo tutto insieme e otteniamo il seguente codice:
console.log(getFriendly(9000));
function getFriendly(range) {
let sums = getAllSum(range);
let res = [];
for (let i = 0; i < sums.length; i++) {
for (let j = i; j < sums.length; j++) {
let sum1 = sums[i];
let sum2 = sums[j];
let num1 = i + 1;
let num2 = j + 1;
if (num1 == sum2 && num2 == sum1) {
res.push([num1, num2]);
}
}
}
return res;
}
function getAllSum(range) {
let arr = [];
for (let i = 1; i <= range; i++) {
arr.push(getOwnDivisorsSum(i));
}
return arr;
}
function getOwnDivisorsSum(num) {
let sum = 0;
for (let i = 1; i < num; i++) {
if (num % i === 0) {
sum += i;
}
}
return sum;
}
Il seguente codice trova i numeri coprimi in un intervallo prefissato. Ottimizzatelo:
console.log(getRelativelyPrime(10000));
function getRelativelyPrime(range) {
let res = [];
for (let i = 2; i <= range; i++) {
for (let j = 2; j < range; j++) {
if (isRelativelyPrime(i, j)) {
res.push([i, j]);
}
}
}
return res;
}
function isRelativelyPrime(num1, num2) {
let arr1 = getDivisors(num1);
let arr2 = getDivisors(num2);
let int = getIntersect(arr1, arr2);
if (int.length === 0) {
return true;
} else {
return false;
}
}
function getIntersect(arr1, arr2) {
let result = [];
for (let elem of arr1) {
if (arr2.includes(elem)) {
result.push(elem);
}
}
return result;
}
function getDivisors(num) {
let res = [];
for (let i = 2; i <= num; i++) {
if (num % i === 0) {
res.push(i);
}
}
return res;
}