【51Nod】1005 大数加法
给出2个大整数A,B,计算A+B的结果。
Input
第1行:大数A 第2行:大数B (A,B的长度 <= 10000 需注意:A B有可能为负数)
Output
输出A + B
Input示例
68932147586 468711654886
Output示例
537643802472
====================================================================================================
问题解法分析:
因为长度小于10000 所以可以依据模拟进位的方法进行解决
解决思路如下:
1:比较两个字符串类型:
如果为如果同为正数(负数) 则进行直接相加(负数相加负号)
若为一正一负,则对负数取消负号进行正数与负数的相减
保证此时字符串已经不存在正负号问题
2:对字符串进行转换
比较两个字符串的长度,若长短不一样则对短的字符串进行补齐 保证两个字符串长短一样
转换函数如下:
void Trans(char *str_num1, char *str_num2, char *tempbuf1, char *tempbuf2)
{
int len_num1=;
int len_num2=;
int i=;
while(str_num1[i]!='\0')
{
len_num1++;
i++;
}
// printf("字符串1的长度: length1=%d\n",len_num1);
i=;
while(str_num2[i]!='\0')
{
len_num2++;
i++;
}
// printf("字符串2的长度: length2=%d\n\n",len_num2); tempbuf1[]='';
tempbuf2[]=''; //=======================================================================
if(len_num2>=len_num1) //补成相同长度
{
for(i=;i<=(len_num2-len_num1);i++)
{
tempbuf1[i]='';
}
for(i=len_num2-len_num1+;i<=len_num2;i++)
{
tempbuf1[i]=str_num1[i-(len_num2-len_num1+)];
}
for(i=;i<=len_num2;i++)
{
tempbuf2[i]=str_num2[i-];
}
}
//------------------------------------------
else if(len_num2<len_num1)
{
for(i=;i<=(len_num1-len_num2);i++)
{
tempbuf2[i]='';
}
for(i=len_num1-len_num2+;i<=len_num1;i++)
{
tempbuf2[i]=str_num2[i-(len_num1-len_num2+)];
}
for(i=;i<=len_num1;i++)
{
tempbuf1[i]=str_num1[i-];
}
} }
3:进行加减运算
对已经处理过的字符串进行加减计算
计算方式模仿竖式进位进行计算
加法算法:
int i=;
int temp=;
int jinwei=;
int len=;
while(tempbuf1[i]!='\0')
{
len++;
i++;
}
for(i=len-;i>=;i--)
{
temp=(int)(tempbuf1[i]+tempbuf2[i]+jinwei-);
if(temp>=)
{
temp=temp-;
jinwei=;
}
else jinwei=;
result[i]=(char)(temp+);
}
减法算法:
int i=;
int temp=;
int jiewei=;
int len=;
int ret=; while(tempbuf1[i]!='\0') // tempbuf1 和 tempbuf2 的长度相等
{
len++;
i++;
} ret = Compare(tempbuf1,tempbuf2);
if(ret==)
{
for(i=len-;i>=;i--)
{
temp = (int)tempbuf1[i] - (int)tempbuf2[i] - jiewei;
if (temp>=)
{
result[i]=(char)(temp+);
jiewei=;
}
else if (temp<)
{
result[i]=(char)(temp++);
jiewei=;
}
}
ThrowAway_0 (result);
}
else if(ret==)
{
memset(result,,);
result[]='';
}
else if(ret==-)
{
for(i=len-;i>=;i--)
{
temp = (int)tempbuf2[i] - (int)tempbuf1[i] - jiewei;
if (temp>=)
{
result[i]=(char)(temp+);
jiewei=;
}
else if (temp<)
{
result[i]=(char)(temp++);
jiewei=;
}
}
ThrowAway_0 (result);
memset(buf1,,);
sprintf(buf1,"-%s",result);
strcpy(result,buf1);
}
在对减法进行运算时,存在大数减小数的问题,因此需要对字符串长度进行比较:
比较函数如下:
int Compare(char *tempbuf1,char *tempbuf2)
{
ThrowAway_0 (tempbuf1);
ThrowAway_0 (tempbuf2);
char buf1[]={};
char buf2[]={};
Trans(tempbuf1,tempbuf2,buf1,buf2);
memset(tempbuf1,,);
memset(tempbuf2,,);
strcpy(tempbuf1,buf1);
strcpy(tempbuf2,buf2); int ret=;
int count=;
while(count<)
{ int m=(int)tempbuf1[count]-;
int n=(int)tempbuf2[count]-;
if(m==n)
{
count++;
ret=;
}
else if(m>n)
{
// printf("tempbuf1>tempbuf2\n");
ret=;
break; }
else if(m<n)
{
// printf("tempbuf1<tempbuf2\n");
ret=-;
break;
}
}
return ret;
}
4:对多余项进行缩减:
将剩余0去掉
void ThrowAway_0 (char *tempbuf ) // 去除结果前面的 连续的无意义的 "0"
{
char buf[]={}; int n = strlen(tempbuf)-;
int i=;
while(i<n)
{
if (tempbuf[i]!='')
{
break;
}
else
{
i++;
}
}
int Throw = i;
for (i=;i<=n-Throw;i++)
{
buf[i]=tempbuf[i+Throw];
} strcpy(tempbuf,buf); }
5:对结果进行输出 完成。
测试代码如下:
#include<stdio.h>
#include<string.h>
#include<stdlib.h> void Add (char *tempbuf1, char *tempbuf2, char *result); // 大数相加 result = tempbuf1 + tempbuf2 void Sub(char *tempbuf1, char *tempbuf2, char *result); // 大数相减 result = tempbuf1 - tempbuf2 (默认前者大) void Trans (char *str_num1, char *str_num2, char *tempbuf1, char *tempbuf2); // 大数转化为等长,且最前面添加 “0” int Compare(char *tempbuf1,char *tempbuf2);//比较两个字符串的长度 相同返回0 buf1长返回1 buf2长返回-1 void ThrowAway_0 (char *tempbuf ); // 去除结果前面的 连续的无意义的 "0" int main()
{ char str1[100001];
char str2[100001];
char result[100001]={0};
short result_flag = 0; scanf("%s %s",str1,str2); if(str1[0] == '-' && str2[0] == '-')
{
str2[0] = str1[0] = '0';
Add(str1,str2,result);
if(result[0]!= '0')
{
printf("-");
printf("%s\n",result);
} else
printf("0\n");
}
else if(str1[0] == '-' && str2[0] != '-')
{
str1[0] ='0'; Sub(str2,str1,result);
printf("%s\n",result);
}
else if(str1[0] != '-' && str2[0] == '-')
{
str2[0] = '0';
Sub(str1,str2,result);
printf("%s\n",result);
}
else
{
Add(str1,str2,result);
printf("%s\n",result);
} return 0;
} void Trans(char *str_num1, char *str_num2, char *tempbuf1, char *tempbuf2)
{
int len_num1=0;
int len_num2=0;
int i=0;
while(str_num1[i]!='\0')
{
len_num1++;
i++;
}
i=0;
while(str_num2[i]!='\0')
{
len_num2++;
i++;
} tempbuf1[0]='0';
tempbuf2[0]='0'; //=======================================================================
if(len_num2>=len_num1) //补成相同长度
{
for(i=1;i<=(len_num2-len_num1);i++)
{
tempbuf1[i]='0';
}
for(i=len_num2-len_num1+1;i<=len_num2;i++)
{
tempbuf1[i]=str_num1[i-(len_num2-len_num1+1)];
}
for(i=1;i<=len_num2;i++)
{
tempbuf2[i]=str_num2[i-1];
}
}
//------------------------------------------
else if(len_num2<len_num1)
{
for(i=1;i<=(len_num1-len_num2);i++)
{
tempbuf2[i]='0';
}
for(i=len_num1-len_num2+1;i<=len_num1;i++)
{
tempbuf2[i]=str_num2[i-(len_num1-len_num2+1)];
}
for(i=1;i<=len_num1;i++)
{
tempbuf1[i]=str_num1[i-1];
}
} } void Add(char *tempbuf1, char *tempbuf2, char *result) // 大数相加 result = tempbuf1 + tempbuf2
{
char buf1[100001]={0};
char buf2[100001]={0};
Trans(tempbuf1,tempbuf2,buf1,buf2);
strcpy(tempbuf1,buf1);
strcpy(tempbuf2,buf2); int i=0;
int temp=0;
int jinwei=0;
int len=0;
while(tempbuf1[i]!='\0')
{
len++;
i++;
}
for(i=len-1;i>=0;i--)
{
temp=(int)(tempbuf1[i]+tempbuf2[i]+jinwei-96);
if(temp>=10)
{
temp=temp-10;
jinwei=1;
}
else jinwei=0;
result[i]=(char)(temp+48);
}
ThrowAway_0 (result); } //================================================ void ThrowAway_0 (char *tempbuf ) // 去除结果前面的 连续的无意义的 "0"
{
char buf[100000]={0}; int n = strlen(tempbuf)-1;
int i=0;
while(i<n)
{
if (tempbuf[i]!='0')
{
break;
}
else
{
i++;
}
}
int Throw = i;
for (i=0;i<=n-Throw;i++)
{
buf[i]=tempbuf[i+Throw];
} strcpy(tempbuf,buf); } //======================================================= //extern "C" __declspec(dllexport)
void Sub(char *tempbuf1, char *tempbuf2, char *result) // 大数相减 result = tempbuf1 - tempbuf2
{
ThrowAway_0 (tempbuf1);
ThrowAway_0 (tempbuf2);
memset(result,0,200);
char buf1[100001]={0};
char buf2[100001]={0};
Trans(tempbuf1,tempbuf2,buf1,buf2);
memset(tempbuf1,0,100001);
memset(tempbuf2,0,100001); strcpy(tempbuf1,buf1);
strcpy(tempbuf2,buf2); int i=0;
int temp=0;
int jiewei=0;
int len=0;
int ret=1; while(tempbuf1[i]!='\0') // tempbuf1 和 tempbuf2 的长度相等
{
len++;
i++;
} ret = Compare(tempbuf1,tempbuf2);
if(ret==1)
{
for(i=len-1;i>=0;i--)
{
temp = (int)tempbuf1[i] - (int)tempbuf2[i] - jiewei;
if (temp>=0)
{
result[i]=(char)(temp+48);
jiewei=0;
}
else if (temp<0)
{
result[i]=(char)(temp+10+48);
jiewei=1;
}
}
ThrowAway_0 (result);
}
else if(ret==0)
{
memset(result,0,100001);
result[0]='0';
}
else if(ret==-1)
{
for(i=len-1;i>=0;i--)
{
temp = (int)tempbuf2[i] - (int)tempbuf1[i] - jiewei;
if (temp>=0)
{
result[i]=(char)(temp+48);
jiewei=0;
}
else if (temp<0)
{
result[i]=(char)(temp+10+48);
jiewei=1;
}
}
ThrowAway_0 (result);
memset(buf1,0,100001);
sprintf(buf1,"-%s",result);
strcpy(result,buf1);
} } //====================================================================== //=================================================================================== int Compare(char *tempbuf1,char *tempbuf2)
{
ThrowAway_0 (tempbuf1);
ThrowAway_0 (tempbuf2);
char buf1[100001]={0};
char buf2[100001]={0};
Trans(tempbuf1,tempbuf2,buf1,buf2);
memset(tempbuf1,0,100001);
memset(tempbuf2,0,100001);
strcpy(tempbuf1,buf1);
strcpy(tempbuf2,buf2); int ret=1;
int count=0;
while(count<100001)
{ int m=(int)tempbuf1[count]-48;
int n=(int)tempbuf2[count]-48;
if(m==n)
{
count++;
ret=0;
}
else if(m>n)
{
// printf("tempbuf1>tempbuf2\n");
ret=1;
break; }
else if(m<n)
{
// printf("tempbuf1<tempbuf2\n");
ret=-1;
break;
}
}
return ret;
}
此外,对于大数运算不限于加减运算 更多运算方法可以参考大数模板:
参考内容:
【51Nod】1005 大数加法的更多相关文章
- 大数高精度加减乘除 51nod 1005 大数加法
1005 大数加法 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 给出2个大整数A,B,计算A+B的结果. Input 第1行:大数A 第2行:大数B ...
- 51nod 1005 大数加法
#include<iostream> #include<string> using namespace std; #define MAXN 10001 },b[MAXN]={} ...
- 51 Nod 1005 大数加法【Java大数乱搞,python大数乱搞】
1005 大数加法 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 给出2个大整数A,B,计算A+B的结果. Input 第1行:大数A 第2行:大数B (A,B的长度 ...
- 51nod 1005 1027 1029 高精度
Java大数用法参考:https://www.cnblogs.com/jin-nuo/p/5313205.html 1005 大数加法: import java.util.*; import java ...
- 51NOD 1005
1005 大数加法 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 给出2个大整数A,B,计算A+B的结果. Input 第1行:大数A 第2行:大 ...
- 51NOD 大数加法以及python写法
练练 大数加法一般为小学生式的"竖式计算"要特别注意的是借位与进位的问题(先给看c++写法,我怕先看了python写法,会看不下去c++写法)这题还有要注意的是 1.同符号的话,直 ...
- c#大数加法
在C#中,我们经常需要表示整数.但是,c#的基本数据类型中,最大的long也只能表示-9,223,372,036,854,775,808 到 9,223,372,036,854,775,807之间的数 ...
- 玲珑杯1007-A 八进制大数加法(实现逻辑陷阱与题目套路)
题目连接:http://www.ifrog.cc/acm/problem/1056 DESCRIPTION Two octal number integers a, b are given, and ...
- 51nod 1027大数乘法
题目链接:51nod 1027大数乘法 直接模板了. #include<cstdio> #include<cstring> using namespace std; ; ; ; ...
随机推荐
- [转载]Android中WebView自适应屏幕
webview中右下角的缩放按钮能不能去掉 settings.setDisplayZoomControls(false); //隐藏webview缩放按钮 让Webview加载的页面居中显示有我知道的 ...
- Nexus入门指南(图文)
Nexus入门指南(图文) 博客分类: Maven JavamavenGoogleApacheTomcat Nexus介绍 Nexus 是Maven仓库管理器,如果你使用Maven,你可以从Maven ...
- 基于Jquery 简单实用的弹出提示框
基于Jquery 简单实用的弹出提示框 引言: 原生的 alert 样子看起来很粗暴,网上也有一大堆相关的插件,但是基本上都是大而全,仅仅几句话可以实现的东西,可能要引入好几十k的文件,所以话了点时间 ...
- [转载]Nginx 反向代理、负载均衡、页面缓存、URL重写及读写分离详解
大纲 一.前言 二.环境准备 三.安装与配置Nginx 四.Nginx之反向代理 五.Nginx之负载均衡 六.Nginx之页面缓存 七.Nginx之URL重写 八.Nginx之读写分离 注,操作系统 ...
- 【分享】改变未来的九大算法[pdf][清晰扫描版]
[下载地址]http://www.colafile.com/file/1179688 图书信息:中文名: 改变未来的九大算法作者: 约翰·麦考密克译者: 管策图书分类: 软件资源格式: PDF版本: ...
- 一个ERP系统的磕磕碰碰
历时5个月的开发,终于可以交付使用了.在这期间系统从需求分析.系统架构设计.开发.测试.运行.迭代.交付,整个忐忑波荡的一路走来,终于有时间静下心来写一篇总结跟各位博友一起探讨.学习了.如果对大家有帮 ...
- iOS基础 - Quartz 2D绘图的基本步骤
一.使用Quartz 2D绘图的基本步骤 1) 获取上下文context(绘制图形的地方) 2) 设置路径(路径是用来描述形状的) 3) 将路径添加到上下文 4) 设置上下文属性(设置颜色,线宽, ...
- Android系统源码导入到eclipse
1.把eclipse工程配置文件复制到android源码根目录下 cp development/ide/eclipse/.classpath ./ 2. 在android源码根目录下新建文 ...
- lnmp1.0 升级php.5.4.28 后出错 Nginx 502 Bad Gateway
碰到一个很奇怪的问题,用lnmp自带的./upgrade_php.sh升级 php5.4.27正常.但升级到php5.4.28就出错,访问p.php 提示:Nginx 502 Bad Gateway. ...
- Private和Protected方法
.NET中如何测试Private和Protected方法 TDD是1)写测试2)写通过这些测试的代码,3)然后重构的实践.在,NET社区中, 这个概念逐渐变得非常流行,这归功于它所增加的质量保证. ...