Dalam matematik pengiraan, pendaraban nombor besar dengan cekap adalah asas kepada pelbagai aplikasi, daripada kriptografi kepada pengkomputeran saintifik. Algoritma pendaraban Karatsuba ialah kaedah bahagi-dan-takluk yang meningkatkan prestasi dengan ketara berbanding pendaraban panjang tradisional untuk nombor besar. Dalam artikel ini, kami akan meneroka pelaksanaan JavaScript bagi algoritma berkuasa ini yang direka untuk mengendalikan nombor besar secara sewenang-wenangnya yang diwakili sebagai rentetan.
Kaedah pendaraban "buku sekolah" standard mempunyai kerumitan masa (O(n2)) , di mana (n) ialah bilangan digit dalam nombor yang didarab. Pertumbuhan kuadratik ini menjadi mahal secara pengiraan apabila bilangannya semakin besar. Algoritma Karatsuba, yang diperkenalkan oleh Anatolii Karatsuba pada tahun 1960, mengurangkan kerumitan ini kepada lebih kurang (O(n1.585)) , menjadikannya pilihan yang lebih pantas untuk input yang besar.
Algoritma bergantung pada strategi bahagi-dan-takluk:
Pendekatan ini mengurangkan bilangan pendaraban rekursif daripada empat kepada tiga, meningkatkan kecekapan.
Di bawah ialah pelaksanaan teguh algoritma Karatsuba dalam JavaScript. Versi ini menyokong integer yang besar sewenang-wenangnya dengan mewakilinya sebagai rentetan.
darab.js
/** * Karatsuba multiplication algorithm for large numbers. * @param {string} num1 - First large number as a string. * @param {string} num2 - Second large number as a string. * @returns {string} - Product of the two numbers as a string. */ function karatsubaMultiply(num1, num2) { // Remove leading zeros num1 = num1.replace(/^0+/, "") || "0"; num2 = num2.replace(/^0+/, "") || "0"; // If either number is zero, return "0" if (num1 === "0" || num2 === "0") return "0"; // Base case for small numbers (12), use Number for safe multiplication if (num1.length <= 12 && num2.length <= 12) { return (Number(num1) * Number(num2)).toString(); } // Ensure even length by padding const maxLen = Math.max(num1.length, num2.length); const paddedLen = Math.ceil(maxLen / 2) * 2; num1 = num1.padStart(paddedLen, "0"); num2 = num2.padStart(paddedLen, "0"); const mid = paddedLen / 2; // Split the numbers into two halves const high1 = num1.slice(0, -mid); const low1 = num1.slice(-mid); const high2 = num2.slice(0, -mid); const low2 = num2.slice(-mid); // Helper function for adding large numbers as strings function addLargeNumbers(a, b) { const maxLength = Math.max(a.length, b.length); a = a.padStart(maxLength, "0"); b = b.padStart(maxLength, "0"); let result = ""; let carry = 0; for (let i = maxLength - 1; i >= 0; i--) { const sum = parseInt(a[i]) + parseInt(b[i]) + carry; result = (sum % 10) + result; carry = Math.floor(sum / 10); } if (carry > 0) { result = carry + result; } return result.replace(/^0+/, "") || "0"; } // Helper function to multiply by 10^n function multiplyByPowerOf10(num, power) { return num === "0" ? "0" : num + "0".repeat(power); } // Helper function for subtracting large numbers function subtractLargeNumbers(a, b) { const maxLength = Math.max(a.length, b.length); a = a.padStart(maxLength, "0"); b = b.padStart(maxLength, "0"); let result = ""; let borrow = 0; for (let i = maxLength - 1; i >= 0; i--) { let diff = parseInt(a[i]) - parseInt(b[i]) - borrow; if (diff < 0) { diff += 10; borrow = 1; } else { borrow = 0; } result = diff + result; } return result.replace(/^0+/, "") || "0"; } // Recursive steps const z0 = karatsubaMultiply(low1, low2); const z1 = karatsubaMultiply( addLargeNumbers(low1, high1), addLargeNumbers(low2, high2) ); const z2 = karatsubaMultiply(high1, high2); // Compute the result using Karatsuba formula const z1MinusZ2MinusZ0 = subtractLargeNumbers( subtractLargeNumbers(z1, z2), z0 ); const powerMidTerm = multiplyByPowerOf10(z1MinusZ2MinusZ0, mid); const z2Term = multiplyByPowerOf10(z2, 2 * mid); // Add all terms const term1 = addLargeNumbers(z2Term, powerMidTerm); const result = addLargeNumbers(term1, z0); return result; } // Example Usage const num1 = "1234567890123456789023454353453454354345435345435435"; const num2 = "98765432109876543210"; console.log("Product:", karatsubaMultiply(num1, num2));
node multiply.js
Pengoptimuman Kes Asas:
Manipulasi Rentetan untuk Ketepatan Arbitrari:
Fungsi Pembantu:
Reka Bentuk Rekursif:
Algoritma Karatsuba mengurangkan bilangan pendaraban rekursif daripada (O(n2)) kepada lebih kurang (O(n1.585)) . Ini menjadikannya lebih pantas daripada kaedah tradisional untuk input besar. Walau bagaimanapun, overhed manipulasi rentetan boleh menjejaskan prestasi untuk input yang lebih kecil, itulah sebabnya pengoptimuman kes asas adalah penting.
Untuk:
/** * Karatsuba multiplication algorithm for large numbers. * @param {string} num1 - First large number as a string. * @param {string} num2 - Second large number as a string. * @returns {string} - Product of the two numbers as a string. */ function karatsubaMultiply(num1, num2) { // Remove leading zeros num1 = num1.replace(/^0+/, "") || "0"; num2 = num2.replace(/^0+/, "") || "0"; // If either number is zero, return "0" if (num1 === "0" || num2 === "0") return "0"; // Base case for small numbers (12), use Number for safe multiplication if (num1.length <= 12 && num2.length <= 12) { return (Number(num1) * Number(num2)).toString(); } // Ensure even length by padding const maxLen = Math.max(num1.length, num2.length); const paddedLen = Math.ceil(maxLen / 2) * 2; num1 = num1.padStart(paddedLen, "0"); num2 = num2.padStart(paddedLen, "0"); const mid = paddedLen / 2; // Split the numbers into two halves const high1 = num1.slice(0, -mid); const low1 = num1.slice(-mid); const high2 = num2.slice(0, -mid); const low2 = num2.slice(-mid); // Helper function for adding large numbers as strings function addLargeNumbers(a, b) { const maxLength = Math.max(a.length, b.length); a = a.padStart(maxLength, "0"); b = b.padStart(maxLength, "0"); let result = ""; let carry = 0; for (let i = maxLength - 1; i >= 0; i--) { const sum = parseInt(a[i]) + parseInt(b[i]) + carry; result = (sum % 10) + result; carry = Math.floor(sum / 10); } if (carry > 0) { result = carry + result; } return result.replace(/^0+/, "") || "0"; } // Helper function to multiply by 10^n function multiplyByPowerOf10(num, power) { return num === "0" ? "0" : num + "0".repeat(power); } // Helper function for subtracting large numbers function subtractLargeNumbers(a, b) { const maxLength = Math.max(a.length, b.length); a = a.padStart(maxLength, "0"); b = b.padStart(maxLength, "0"); let result = ""; let borrow = 0; for (let i = maxLength - 1; i >= 0; i--) { let diff = parseInt(a[i]) - parseInt(b[i]) - borrow; if (diff < 0) { diff += 10; borrow = 1; } else { borrow = 0; } result = diff + result; } return result.replace(/^0+/, "") || "0"; } // Recursive steps const z0 = karatsubaMultiply(low1, low2); const z1 = karatsubaMultiply( addLargeNumbers(low1, high1), addLargeNumbers(low2, high2) ); const z2 = karatsubaMultiply(high1, high2); // Compute the result using Karatsuba formula const z1MinusZ2MinusZ0 = subtractLargeNumbers( subtractLargeNumbers(z1, z2), z0 ); const powerMidTerm = multiplyByPowerOf10(z1MinusZ2MinusZ0, mid); const z2Term = multiplyByPowerOf10(z2, 2 * mid); // Add all terms const term1 = addLargeNumbers(z2Term, powerMidTerm); const result = addLargeNumbers(term1, z0); return result; } // Example Usage const num1 = "1234567890123456789023454353453454354345435345435435"; const num2 = "98765432109876543210"; console.log("Product:", karatsubaMultiply(num1, num2));
Hasilnya ialah:
node multiply.js
Algoritma pendaraban Karatsuba ialah penyelesaian yang praktikal dan cekap untuk mendarab nombor besar. Pelaksanaan ini menunjukkan kuasa dan fleksibilitinya apabila mengendalikan input besar secara sewenang-wenangnya dalam JavaScript. Dengan keperluan yang semakin meningkat untuk aritmetik berketepatan tinggi, menguasai algoritma sedemikian boleh meningkatkan keupayaan pengiraan dalam pelbagai aplikasi.
Atas ialah kandungan terperinci Memahami dan Melaksanakan Algoritma Pendaraban Karatsuba untuk Nombor Besar. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!