nowcoder NC10 大数乘法

题目链接: icon-default.png?t=N7T8https://www.nowcoder.com/practice/c4c488d4d40d4c4e9824c3650f7d5571?tpId=196&tqId=37177&rp=1&ru=/exam/company&qru=/exam/company&sourceUrl=%2Fexam%2Fcompany&difficulty=undefined&judgeStatus=undefined&tags=&title=

 

题目描述:

以字符串的形式读入两个数字,编写一个函数计算它们的乘积,以字符串形式返回。

数据范围: 读入的数字大小满足 0 \leqslant n \leqslant {10}^{1000}

要求:空间复杂度 O(m),时间复杂度 O(m^{2})(假设m是n的长度)

示例1:

输入:"11","99"

返回值:"1089"

说明:11*99=1089

示例2:

输入:"1","0"

返回值:"0"

答案:

import java.util.*;public class Solution {public String solve (String s, String t) {// write code hereif (s.charAt(0) == '0' || t.charAt(0) == '0'){return "0";}String ret = "0";String[] tmp = new String[t.length()]; for (int i = t.length() - 1; i >= 0; i--){tmp[i] = "";int j = t.length() - 1;while(j - i > 0){tmp[i] += '0';j--;}tmp[i] += alongMultiply(s, t.charAt(i));}for (int i = t.length() - 1; i >= 0; i--){ret = Add(tmp[i], ret);}StringBuffer stringBuffer = new StringBuffer();for (int i = ret.length() - 1; i >= 0; i--){stringBuffer.append(ret.charAt(i));}ret = stringBuffer.toString();return ret;}public String Add(String a, String b){String str = "";int aLen = a.length() - 1;int ai = 0;int bLen = b.length() - 1;int bi = 0;int ten = 0;while(aLen >= ai && bLen >= bi){int tmp = (a.charAt(ai++) - '0') + (b.charAt(bi++) - '0');tmp += ten;ten = tmp / 10;str += tmp % 10;}while(aLen >= ai){int tmp = a.charAt(ai++) - '0';tmp += ten;ten = tmp / 10;str += tmp % 10;}while(bLen >= bi){int tmp = b.charAt(bi++) - '0';tmp += ten;ten = tmp / 10;str += tmp % 10;}if (ten != 0){str += ten;}return str;}public String alongMultiply(String s, char t){String ret = "";if (s.charAt(0) == '0' || t == '0'){return "0";}int tt = t - '0';int ten = 0;for (int i = s.length() - 1; i >= 0; i--){int tmp = s.charAt(i) - '0';tmp *= tt;tmp += ten;ten = tmp / 10;ret += tmp % 10;}if (ten != 0){ret += ten;}return ret;}
}

详解: 

 从题目中我们可以得到以下几点信息:

  1. 输入值和返回值都是字符串类型;
  2. 输入值和返回值不可以直接转换成整数(因为数字过大);
  3. 对时间复杂度几乎没有要求;
  4. 不会出现负数乘法。

 当我们清楚了题目要求之后就该考虑该如何解题了。

 首先我们应该考虑的是乘法是如何进行计算的

我们以 11 * 99 为例:

 

我们可以分析得到无论是几位数的乘法都是按照以下步骤进行的:

  1. 将第一个数字分别乘以第二个数的每一位;
  2. 如果第一个数乘的是第二个数的个位就给结果乘一,十位就乘十 以此类推;
  3. 最后一步就是将各各结果相加。

到这一步之后如果你想在题目给的那一个方法里面实现这些内容就会大大提高你写代码的难度,此时其实我推荐将其用三个方法来实现。

  • 第一个为主函数,主要用来实现代码的整体思路;
  • 第二个为相乘的方法,其主要功能是实现一个 n 位数与一位数相乘;
  • 第三个为相加的方法,其主要功能是实现两个 n 位数的相加。

根据乘法的定义可以知道:0 与任何数相乘都是 0 所以我们的第一段代码就可以为:

    public String solve (String s, String t) {// write code hereif (s.charAt(0) == '0' || t.charAt(0) == '0'){return "0";}}

接下来就是对每一位进行相乘但是我们并不知道是 几位数与几位数进行相乘 所以我们此时应该根据 t 的长度来定义一个字符串数组 tmp 用来存储 t 中的每一位与 s 相乘的结果。

再定义一个名为 alongMultiply() 的方法 此方法就用来实现n 位数与一位数相乘并将其值以字符串的形式进行返回。(此方法可以先不急着实现)。

再定义一个名为 的字符串类型的变量,将其初始化为“0” 用来存储最终的返回值。

因为会有进位而导致最后结果的位数充满不确定性所以我们可以采用倒序的存储方式 

即:12345 存储为 54321

因为加法也会有进位所以我们可以在主方法的最后统一进行反转。

 public String solve (String s, String t) {// write code hereif (s.charAt(0) == '0' || t.charAt(0) == '0'){return "0";}String ret = "0";//新加的代码String[] tmp = new String[t.length()]; for (int i = t.length() - 1; i >= 0; i--){tmp[i] = "";int j = t.length() - 1;while(j - i > 0){ //相当于十位乘十 , 百位乘一百……tmp[i] += '0';j--;}tmp[i] += alongMultiply(s, t.charAt(i));}
}

紧接着我们再将 tmp数组 中的所有值进行相加存储在 ret 中。

for (int i = t.length() - 1; i >= 0; i--){ret = Add(tmp[i], ret);
}

到这里我们的整体布局已经完成了,接下来就该实现 alongMultiply() 方法了:

public String alongMultiply(String s, char t){String ret = ""; //用来存储最后的返回值if (s.charAt(0) == '0' || t == '0'){return "0";}int tt = t - '0';int ten = 0; //用来存储每次的进位for (int i = s.length() - 1; i >= 0; i--){int tmp = s.charAt(i) - '0';tmp *= tt;tmp += ten;ten = tmp / 10;ret += tmp % 10;}if (ten != 0){ret += ten;}return ret;}

 Add() 方法的实现:

public String Add(String a, String b){String str = ""; //存储最终的返回值int aLen = a.length() - 1;int ai = 0;int bLen = b.length() - 1;int bi = 0;int ten = 0; //用来存储每次的进位while(aLen >= ai && bLen >= bi){int tmp = (a.charAt(ai++) - '0') + (b.charAt(bi++) - '0');tmp += ten;ten = tmp / 10;str += tmp % 10;}while(aLen >= ai){int tmp = a.charAt(ai++) - '0';tmp += ten;ten = tmp / 10;str += tmp % 10;}while(bLen >= bi){int tmp = b.charAt(bi++) - '0';tmp += ten;ten = tmp / 10;str += tmp % 10;}if (ten != 0){str += ten;}return str;}

接下来我们只要再将最终的值进行反转本题就算做完了:

        StringBuffer stringBuffer = new StringBuffer();for (int i = ret.length() - 1; i >= 0; i--){stringBuffer.append(ret.charAt(i));}ret = stringBuffer.toString();

 当然你也可以用(我用上面的方法主要是因为它比较快):

        tmp[0] = ret;ret = "";for (int i = tmp[0].length() - 1; i >= 0; i--){ret += tmp[0].charAt(i);}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/71517.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

222. 完全二叉树的节点个数

题目链接: 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台 我的想法: 递归法 万金油--层次遍历法 当然上面两中都是笨方法,就算不是完全二叉树也能算,没有用到完全二叉树的特性。 我的代码&#xff1…

linux线程讲解

1.线程概述 一个进程在同一时刻只做一件事情,进程是程序执行的一个实例。 线程是操作系统能够进行运算调度的最小单位,一个进程中可以并发多个线程,每条线程并行执行不同的任务。 进程:资源分配的最小单位。线程,程…

50etf期权最多能开仓多少手?

50etf期权限仓限额的操作,是为了能更好防范和控制期权交易的风险,无论是期货还是期权,在交易中都有规定的持仓限额,不能超过某个额度,那么50etf期权最多能开仓多少手?下文为你们全面介绍!本文来…

【数据结构】单链表详解

当我们学完顺序表的时候,我们发现了好多问题如下: 中间/头部的插入删除,时间复杂度为O(N)增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当…

纯手工总结超详细关于计算机网络的五层知识点,看看你都掌握了没

纯手工总结超详细关于计算机网络的五层知识点,看看你都掌握了没 文章目录 纯手工总结超详细关于计算机网络的五层知识点,看看你都掌握了没1.应用层1.1 HTTP协议1.1.1 URL1.1.2 HTTP方法1.1.3 HTTP请求1.1.4 HTTP状态码1.1.5 HTTP会话保持 1.2 HTTPS协议 …

Linux上安装FTP

1、登录FTP,执行安装命令 yum -y install vsftpd 2、启动FTP服务器,设置开启自启动 systemctl enable vsftpd.service systemctl start vsftpd.service systemctl status vsftpd.service #查看状态, 显示active说明FTP启动成功 3、修改FTP配置文件/et…

数据通信——传输层TCP(可靠传输机制的滑动窗口)

引言 之前提到过拥塞问题,如果大量数据疯狂涌入,接收端无法及时处理就会导致数据丢包,从而使得通信受到干扰。之前的连续ARQ如果不加以节制,疯狂发送报文,接收端无法及时返回ACK就会导致网络瘫痪。 滑动窗口机制协议 这…

阿里云部署开源MQTT平台mosquitto的docker操作

MQTT(Message Queuing Telemetry Transport)是一种轻量级的消息传输协议,广泛用于物联网和传感器网络中。Mosquitto是一个流行的开源MQTT代理,可以在Docker中进行配置和部署。本文将详细介绍如何在Docker中配置Mosquitto MQTT代理…

maven打包时显示无效jdk版本

1、配置当前项目所需的Jdk版本 2、与当前项目指定的jdk版本相同 3、与当前项目指定的jdk版本相同 4、与当前项目指定的jdk版本相同 5、指定主项目启动时的vm配置与当前项目所需版本相同

Stable Diffusion — ControlNet 超详细讲解

Stable Diffusion — ControlNet 超详细讲解 ControlNet 最近非常火🔥!ControlNet 和 Stable Diffusion 的结合使 Stable Diffusion 能够接受指导图像生成过程的条件输入,从而增强了 Stable Diffusion 的性能。今天为大家深入剖析 ControlNe…

如何让自己的精力集中 Maven自学笔记 马云演讲观看

目录 如何让自己的精力集中 Avoid having multiple tasks and objects in your line of sight 人的脑袋是给自己思考用的 晚上床上想千条路,早上起床还是走原路 参与才会变得更好 共度灾难,是需要互相鼓励的 CFO Capital 上海各区都有哪些大学?…

git快速查看某个文件修改的所有commit

1. git blame file git blame 可以显示历史修改的每一行记录,有时候我们只想了解某个文件一共提交几次commit,只显示commit列表,这种方式显然不满足要求。 2.git log常规使用 (1)显示整个project的所有commit (2)显示某个文件的所有commit 这是git log不添加参数的常规…

纽扣电池/锂电池UN38.3安全检测报告

根据规章要求,航空公司和机场货物收运部门应对锂电池进行运输文件审查,重要的是每种型号的锂电池UN38.3安全检测报告。该报告可由的三方检测机构。如不能提供此项检测报告,将禁止锂电池进行航空运输. UN38.3包含产品:1、 锂电池2…

无涯教程-JavaScript - BESSELK函数

描述 BESSELK函数返回修改后的Bessel函数Kn(x),该函数等效于针对纯虚参判断的Bessel函数。 这些也称为双曲贝塞尔函数。 语法 BESSELK(X, N)争论 Argument描述Required/OptionalXThe value at which to evaluate the function.RequiredNThe order of the function. If n i…

Matlab进阶绘图第27期—水平双向堆叠图

在上一期文章中,分享了Matlab双向堆叠图的绘制方法: 进一步,再来看一下水平双向堆叠图的绘制方法(由于Matlab中未收录水平双向堆叠图的绘制函数,因此需要大家自行设法解决)。 先来看一下成品效果&#xff…

基于多设计模式下的同步异步日志系统

基于多设计模式下的同步&异步日志系统 代码链接:https://github.com/Janonez/Log_System 1. 项目介绍 本项目主要实现一个日志系统, 其主要支持以下功能: 支持多级别日志消息支持同步日志和异步日志支持可靠写入日志到标准输出、文件…

【Three.js】第二十一章 Physics 物理

介绍 物理是WebGL可以添加到项目体验中最酷的功能之一。人们喜欢真实物理感的物体,看到它们碰撞、倒塌、坠落和弹跳,就像我的作品集一样: https: //bruno-simon.com/ 有很多方法可以将物理功能添加到您的项目中,这取决于您想要实…

21.4 CSS 盒子模型

1. 边框样式 border-style属性: 指定元素的边框样式.常用属性值: - none: 无边框(默认值). - solid: 实线边框. - dotted: 点状边框. - dashed: 虚线边框. - double: 双线边框. - groove: 凹槽状边框. - ridge: 脊状边框. - inset: 内阴影边框. - outset: 外阴影边框.这些值可…

使用SimPowerSystems并网光伏阵列研究(Simulink实现)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…

C++:类和对象(二)

本文主要介绍:构造函数、析构函数、拷贝构造函数、赋值运算符重载、const成员函数、取地址及const取地址操作符重载。 目录 一、类的六个默认成员函数 二、构造函数 1.概念 2.特性 三、析构函数 1.概念 2.特性 四、拷贝构造函数 1.概念 2.特征 五、赋值…