gcd.js

import { isNaN, round, toNumber } from 'ut2';
/**
 * 最大公约数,使用辗转相除法。
 *
 * 遵循以下约定:
 *
 * 1. 如果参数中包含无效数值,返回 `NaN` 。
 * 2. 如果全部参数都为 `0` ,返回 `0` 。
 * 3. 如果参数包含 `0` ,仅计算非零的数。
 * 4. 如果参数为负数,将转为绝对值的正数。
 * 5. 如果参数包含小数点,将转为四舍五入的整数。
 *
 * @alias module:Math.gcd
 * @since 4.20.0
 * @see {@link https://baike.baidu.com/item/最大公约数 最大公约数}
 * @param {...(number|string)} nums 两个或多个整数。
 * @returns {number} 最大公约数。
 * @example
 *
 * gcd(8, 14); // 2
 * gcd(57, 48); // 3
 * gcd(140, 21, 42); // 7
 * gcd('foo', 'bar'); // NaN
 * gcd(0, 10); // 10
 * gcd(2.3, 3.8, 8, -10); // 2
 *
 */
function gcd(...nums) {
    let num1 = nums[0];
    let num2 = nums[1] === void 0 ? 0 : nums[1];
    const rest = nums.slice(2);
    if (rest.length > 0) {
        // eslint-disable-next-line prefer-spread
        return gcd.apply(void 0, [gcd(num1, num2)].concat(rest));
    }
    num1 = Math.abs(round(num1));
    num2 = Math.abs(round(num2));
    // 兼容处理,如果参数包含无效数值时,返回 NaN
    if (isNaN(num1) || isNaN(num2)) {
        return Number.NaN;
    }
    if (num1 === 0 && num2 === 0) {
        return 0;
    }
    if (num1 === 0) {
        return num2;
    }
    if (num2 === 0) {
        return num1;
    }
    let temp = num2;
    if (num1 < num2) {
        temp = num1;
        num1 = num2;
        num2 = temp;
    }
    while (temp) {
        temp = num1 % num2;
        num1 = num2;
        num2 = temp;
    }
    return toNumber(num1);
}
export default gcd;