前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >大整数相乘“分治法”和“循环暴力法”

大整数相乘“分治法”和“循环暴力法”

原创
作者头像
本人秃顶程序员
修改2019-04-22 11:09:26
6580
修改2019-04-22 11:09:26
举报
文章被收录于专栏:Java架构筑基Java架构筑基

前言

今天刷到一道很有趣的面试题,感觉很有意思,来分享给大家。

题目描述

有两个用字符串表示的非常大的大整数,算出他们的乘积,也是用字符串表示。不能用系统自带的大整数类型。

输入描述: 空格分隔的两个字符串,代表输入的两个大整数 输出描述: 输入的乘积,用字符串表示

示例1

输入 72106547548473106236 982161082972751393 输出 70820244829634538040848656466105986748

思路分析

例如x=1234,y=567

  • ①将x拆分成两半儿,a = 12 b = 34
  • ②将y拆分成两半儿,c = 5 d = 67
  • ③则xy = (12102+34)(5102+67) = (a102+b)(c102+d) = ac104+ad102+bc102+bd
  • ④递归求(ac),(ad),(bc),(bd)的结果,如果a,b,c,d足够小,就直接相乘算出结果,否则,从第①步开始重复,继续拆分a,b,c,d,直至到了能直接算结果的时候,递归结束,开始回溯
代码语言:javascript
复制
import java.util.Arrays;
import java.util.Scanner;
public class Main {
  public static void main(String[] args){
      Scanner sca = new Scanner(System.in);
      String x = sca.nextLine();
      String y = sca.nextLine();
      System.out.println(f(x,y));
  }
  
  //分治法
  public static Long f(String x,String y){
      String a = x.substring(0, x.length()/2);
      String b = x.substring(x.length()/2);
      String c = y.substring(0, y.length()/2);
      String d = y.substring(y.length()/2);
      int n = b.length();
      int m = d.length();
      if(x.length()<=4 && y.length()<=4){
          return (long) (Integer.parseInt(x)*Integer.parseInt(y));
      }
      if(x.length()>4 && y.length()<=4){
          return (long) (f(a,y)*Math.pow(10, n)+f(b,y));
      }
      if(y.length()>4 && x.length()<=4){
          return (long) (f(c,x)*Math.pow(10, m)+f(d,x));
      }else{
          return (long) (f(a,c)*Math.pow(10, n+m)+f(a,d)*Math.pow(10, n)+f(b,c)*Math.pow(10, m)+f(b,d));
      }
  }
}

上述思路,时间复杂度是o(log2max(n,m)),其中n是x的长度,m是y的长度,

但是当最后的乘积超过long型的时候,还是会错误,

我一直没想到好的方法完全解决,百度了一下,试了好几个人的java代码,结果都是报错,有的甚至用long型变量接收输入的大整数,直接就报错了,没有一个是对的,访问量还那么高,真水啊,,,,,,

然后想了另一种方法,可以完美解决此问题,时间复杂度是o(n2):

循环暴力法:

  • ①把两个字符串经过拆分转换成int型数组
  • ②用intx[]里的每个数字乘以inty[]里面的每一个数字,就是传统的在纸上手算的那个过程,将结果存入另一个数组
  • ③如果两数相乘是两位数,就把十位上的数加到高位上。

循环结束后,两个大数的乘积就按位数存到数组里了。

这个方法适用于所有的大数相乘。

java 代码如下

代码语言:javascript
复制
import java.util.Arrays;
import java.util.Scanner;
public class Main {
    public static void main(String[] args){
        Scanner sca = new Scanner(System.in);
        String x = sca.nextLine();
        String y = sca.nextLine();
        System.out.println(f(x,y));
    }
    public static String f(String x,String y){
        int[] intx = new int[x.length()];
        int[] inty = new int[y.length()];
        int[] intsum = new int[x.length()+y.length()];

        for (int i = 0; i < x.length(); i++) {
            intx[x.length()-1-i] = Integer.parseInt(x.substring(i, i+1));
        }
        for (int i = 0; i < y.length(); i++) {
            inty[y.length()-1-i] = Integer.parseInt(y.substring(i, i+1));
        }
        for (int i = 0; i < intx.length; i++) {
            for (int j = 0; j < inty.length; j++) {
                intsum[i+j] += intx[i]*inty[j];
            }
            for (int j = 0; j < intsum.length-1; j++) {
                if(intsum[j]>9){
                    intsum[j+1]+=intsum[j]/10;
                    intsum[j] = intsum[j]%10;
                }
            }
        }
        String str = "";
        boolean t = false;
        for (int i = intsum.length-1; i >=0; i--) {
            if(intsum[i]!=0) t = true;
            if(t) str = str+intsum[i];
        }
        return str;
    }
}

希望大家能多多指教!

读者福利:

分享免费学习资料

针对于Java程序员,我这边准备免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)

为什么某些人会一直比你优秀,是因为他本身就很优秀还一直在持续努力变得更优秀,而你是不是还在满足于现状内心在窃喜!希望读到这的您能点个小赞和关注下我,以后还会更新技术干货,谢谢您的支持!

资料领取方式:加入Java技术交流群963944895点击加入群聊,私信管理员即可免费领取

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 题目描述
    • 示例1
    • 思路分析
      • 循环暴力法:
        • java 代码如下
    • 读者福利:
    相关产品与服务
    云数据库 Redis
    腾讯云数据库 Redis(TencentDB for Redis)是腾讯云打造的兼容 Redis 协议的缓存和存储服务。丰富的数据结构能帮助您完成不同类型的业务场景开发。支持主从热备,提供自动容灾切换、数据备份、故障迁移、实例监控、在线扩容、数据回档等全套的数据库服务。
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档