Java 小例子:大数乘法和阶乘

这是一个不借助 BigInteger 来实现大数乘法的例子,顺便做了一个阶乘。

大数乘法的实现是基于印度的格子乘法(参考这里),使用这种方法,计算 m 位数乘以 n 位数只需要创建一个 m+n 位的数组保存结果即可。

当然,这种方式效率绝不可能跟 BigInteger 比,但作为初学者的练习,是很有价值的。

public class BigMultiply {
 
    public static void main(String[] args) {
        System.out.println(factorial("1000"));
    }
 
    /**
     * 阶乘
     *
     * @param n 要进行阶乘的数字
     *
     * @return 阶乘结果
     */
    private static String factorial(String n) {
        if (Integer.parseInt(n) <= 1) {
            return "1";
        } else {
            return bigMultiply(n, factorial(String.valueOf(Integer.parseInt(n) - 1)));
        }
    }
 
    /**
     * 大数字乘法
     *
     * @param number1 数字1
     * @param number2 数字2
     *
     * @return 乘法结果
     */
    private static String bigMultiply(String number1, String number2) {
        short[] numbers1 = toIntArray(number1);
        short[] numbers2 = toIntArray(number2);
 
        short[] result = multiply(numbers1, numbers2);
        return toString(result);
    }
 
    private static short[] multiply(short[] numbers1, short[] numbers2) {
        short[] result = new short[numbers1.length + numbers2.length];
 
        for (int i = 0; i < numbers1.length; i++) {
            for (int j = 0; j < numbers2.length; j++) {
                int tenth = i + j;
                int cellResult = numbers1[i] * numbers2[j];
 
                put(result, tenth + 1, cellResult % 10);
                put(result, tenth, cellResult / 10);
            }
        }
 
        return result;
    }
 
    /**
     * put <i>number</i> into <i>index</i> position of <i>result</i>
     *
     * @param result -
     * @param index  -
     * @param number -
     */
    private static void put(short[] result, int index, int number) {
        result[index] += number;
 
        carryFrom(result, index);
    }
 
    /**
     * 进位
     *
     * @param result 数组
     * @param index  开始进位的位置
     */
    private static void carryFrom(short[] result, int index) {
        if (index < 0) {
            return;
        }
 
        if (result[index] >= 10) {
            result[index - 1]++;
            result[index] = (short)(result[index] - 10);
            carryFrom(result, index - 1);
        }
    }
 
    private static short[] toIntArray(String numbers) {
        short[] result = new short[numbers.length()];
        for (int i = 0; i < numbers.length(); i++) {
            result[i] = (short)(Integer.parseInt(numbers.substring(i, i + 1)));
        }
        return result;
    }
 
    private static String toString(short[] ints) {
        int start = 0;
        while (ints[start] == 0) {
            start++;
        }
 
        String str = "";
        while (start < ints.length) {
            str += ints[start];
            start++;
        }
 
        return str;
    }
}

 

参与评论 您还未登录,请先 登录 后发表或查看评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
©️2022 CSDN 皮肤主题:撸撸猫 设计师:马嘣嘣 返回首页

打赏作者

捏造的信仰

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值