c语言:分治算法之大数相乘
我们把整数A由规模n分为n1和n2,把整数B由规模m分为m1和m2,如下图:


则A分为n1位的A1和n2位的A1,B分为m1位的B1和m2位的B2,如下式所示:

以此类推,我们可以把A1、A2、B1、B2继续划分,直至最小单位。(这里在编程时需要用递归来实现)
上面讲的很清楚了,那么A和B的相乘就可以表示为:

#include <stdio.h>
#include <malloc.h>
#include <stdlib.h>
#include <string.h> char *result = '\0';
int pr = 1; void getFill(char *a,char *b,int ia,int ja,int ib,int jb,int tbool,int move){
int r,m,n,s,j,t;
char *stack;
// get(a,b,0,0,0,0,1,2)
m = a[ia] - 48; // m=1
if( tbool ){// 直接从结果数组的标志位填入,这里用了堆栈思想 这部分实现A1xB1
r = (jb - ib > ja - ia) ? (jb - ib) : (ja - ia); // r=0
stack = (char *)malloc(r + 4); // 因为执行了getFill方法,此时ia=ja,说明两个数必定其中一个是个位数,而jb-ib的值+1即另一个数的位数
for(r = j = 0,s = jb; s >= ib; r ++,s --){ // r=j=s=0;0>=0;r++,s++ r=1,s=-1
n = b[s] - 48; // n=4
stack[r] = (m * n + j) % 10; //stack[0]=(1x4+0)%10=4 通过模10来获取result[1]位置上的数
j = (m * n + j) / 10; // j=(1x4+0)/10 = 0 通过整除来获取result[1]位置上的数,如果j=0,则只需要填写result[1]位置的数;如果j不等于0,则将j放在
} // result[1]位置,前面取模获取的值放在result[2]所在的位置
if( j ){ // 如果j 不等于0,如j=1,stack[1]=1,则将stack[1]赋值给result[1],将stack[0]赋值个result[2].由于先有stack[0],再有stack[1],而使用时先用
stack[r] = j; // stack[1]=1 stack[1]再用stack[0],即先进后出,所以说使用了栈的思想。
r ++; // r=2
}
// r=1
for(r --; r >= 0; r --,pr ++) // r=0;0>=0; pr=2
result[pr] = stack[r]; //result[1]=stack[0]=4
free(stack);
// move=2 pr=2 由于A1xB1之后后面还要加00,故将result[1]后面的两个字节空间赋值为\0.字符串是以\0作为结束标志的,在内存中,比如字符串“abc”,实际上是4个字节的空间abc\0
for(move = move + pr; pr < move; pr ++) // move=4;2<4;pr++ pr=3 move=4;3<4;pr++ pr=4
result[pr] = '\0'; // result[2]='\0' result[3]='\0'
}
else{ //与结果的某几位相加,这里不改变标志位 pr 的值 //这部分实现A2xB1,A1xB2,A2xB2
r = pr - move - 1;
for(s = jb,j = 0; s >= ib; r --,s --){
n = b[s] - 48;
t = m * n + j + result[r];
result[r] = t % 10;
j = t / 10;
}
for( ; j ; r -- ){
t = j + result[r];
result[r] = t % 10;
j = t / 10;
}
}
} int get(char *a,char *b,int ia,int ja,int ib,int jb,int t,int move){
int m,n,s,j; if(ia == ja){
getFill(a,b,ia,ja,ib,jb,t,move);
return 1;
}
else if(ib == jb){
getFill(b,a,ib,jb,ia,ja,t,move);
return 1;
}
else{ // 12x45 get(a,b,0,1,0,1,1,0)
m = (ja + ia) / 2; // m=0
n = (jb + ib) / 2; // m=0
s = ja - m; // s=1
j = jb - n; // j=1
get(a,b,ia,m,ib,n,t,s + j + move); // get(a,b,0,0,0,0,1,2)
get(a,b,ia,m,n + 1,jb,0,s + move); // get(a,b,0,0,1,1,0,1)
get(a,b,m + 1,ja,ib,n,0,j + move); // get(a,b,1,1,0,0,0,1)
get(a,b,m + 1,ja,n + 1,jb,0,0 + move); // get(a,b,1,1,1,1,0,0)
}
return 0;
} int main(){
char *a,*b;
int n,flag; a = (char *)malloc(1000); // 分配所需的内存空间 1000字节
b = (char *)malloc(1000);
printf("The program will computer a*b\n");
printf("Enter a b:");
scanf("%s %s",a,b);
result = (char *)malloc(strlen(a) + strlen(b) + 2);
flag = pr = 1;
result[0] = '\0';
if(a[0] == '-' && b[0] == '-')
get(a,b,1,strlen(a)-1,1,strlen(b)-1,1,0);
if(a[0] == '-' && b[0] != '-'){
flag = 0;
get(a,b,1,strlen(a)-1,0,strlen(b)-1,1,0);
}
if(a[0] != '-' && b[0] == '-'){
flag = 0;
get(a,b,0,strlen(a)-1,1,strlen(b)-1,1,0);
}
if(a[0] != '-' && b[0] != '-')
get(a,b,0,strlen(a)-1,0,strlen(b)-1,1,0); // get(a,b,0,1,0,1,1,0)
if(!flag)
printf("-");
if( result[0] )
printf("%d",result[0]);
for(n = 1; n < pr ; n ++)
printf("%d",result[n]);
printf("\n");
free(a);
free(b);
free(result);
system("pause");
return 0;
}
void getFill(char *a,char *b,int ia,int ja,int ib,int jb,int tbool,int move){
int r,m,n,s,j,t;
char *stack;
// get(a,b,0,0,1,1,0,1) 计算A1xB2
m = a[ia] - 48; // m=1 m为A1的值
if( tbool ){// 直接从结果数组的标志位填入,这里用了堆栈思想
r = (jb - ib > ja - ia) ? (jb - ib) : (ja - ia);
stack = (char *)malloc(r + 4);
for(r = j = 0,s = jb; s >= ib; r ++,s --){
n = b[s] - 48; // n=4
stack[r] = (m * n + j) % 10;
j = (m * n + j) / 10;
}
if( j ){
stack[r] = j;
r ++;
}
for(r --; r >= 0; r --,pr ++)
result[pr] = stack[r];
free(stack);
for(move = move + pr; pr < move; pr ++)
result[pr] = '\0';
}
else{ //与结果的某几位相加,这里不改变标志位 pr 的值
r = pr - move - 1; // pr=4 move=1 r=4-1-1=2
for(s = jb,j = 0; s >= ib; r --,s --){ // s=jb=1,j=0;1>=1;r--,s--
n = b[s] - 48; // n=5 n为B2的值
t = m * n + j + result[r]; // t=1*5+0+result[2]=5+0=5
result[r] = t % 10; // result[2]=5 将两个数相乘的结果模10得到result[2]上的值,
j = t / 10; // j=0 将两个数相乘的结果整除10得到是否需要进一位,
}
// r=1,s=0
for( ; j ; r -- ){
t = j + result[r];
result[r] = t % 10;
j = t / 10;
}
}
}
void getFill(char *a,char *b,int ia,int ja,int ib,int jb,int tbool,int move){
int r,m,n,s,j,t;
char *stack;
// get(a,b,1,1,0,0,0,1) 计算A2xB1
m = a[ia] - 48; // m=2 m为A2的值
if( tbool ){// 直接从结果数组的标志位填入,这里用了堆栈思想
r = (jb - ib > ja - ia) ? (jb - ib) : (ja - ia);
stack = (char *)malloc(r + 4);
for(r = j = 0,s = jb; s >= ib; r ++,s --){
n = b[s] - 48;
stack[r] = (m * n + j) % 10;
j = (m * n + j) / 10;
}
if( j ){
stack[r] = j;
r ++;
}
// r=1
for(r --; r >= 0; r --,pr ++)
result[pr] = stack[r];
free(stack);
for(move = move + pr; pr < move; pr ++)
result[pr] = '\0';
}
else{ //与结果的某几位相加,这里不改变标志位 pr 的值
r = pr - move - 1; // pr=4 move=1 r=4-1-1=2
for(s = jb,j = 0; s >= ib; r --,s --){ // s=jb=0,j=0;0>=0;r--,s--
n = b[s] - 48; // n=4 n为B1的值
t = m * n + j + result[r]; // t=2*4+0+result[2]=8+5=13 两个数相乘的结果加上前面A1xB2的结果
result[r] = t % 10; // result[2]=3 将两个数相乘的结果加上前面A1xB2的结果模10得到最后result[2]的值
j = t / 10; // j=1 r=1 s=-1 整除10看是否需要进一位,如果需要进一位,则将j与原来result[1]相加,result[1]再对10取模,即可确定result[1]的值
}
// r=1,s=-1
for( ; j ; r -- ){
t = j + result[r]; // t=1+4=5
result[r] = t % 10; // result[1]=5
j = t / 10; // j=0 r=0
}
}
}
void getFill(char *a,char *b,int ia,int ja,int ib,int jb,int tbool,int move){
int r,m,n,s,j,t;
char *stack;
// get(a,b,1,1,1,1,0,0) 计算A2xB2
m = a[ia] - 48; // m=2 m为A2的值
if( tbool ){// 直接从结果数组的标志位填入,这里用了堆栈思想
r = (jb - ib > ja - ia) ? (jb - ib) : (ja - ia);
stack = (char *)malloc(r + 4);
for(r = j = 0,s = jb; s >= ib; r ++,s --){
n = b[s] - 48;
stack[r] = (m * n + j) % 10;
j = (m * n + j) / 10;
}
if( j ){
stack[r] = j;
r ++;
}
for(r --; r >= 0; r --,pr ++)
result[pr] = stack[r];
free(stack);
for(move = move + pr; pr < move; pr ++)
result[pr] = '\0';
}
else{ //与结果的某几位相加,这里不改变标志位 pr 的值
r = pr - move - 1; // pr=4 move=0 r=4-0-1=3
for(s = jb,j = 0; s >= ib; r --,s --){ // s=jb=1,j=0;1>=1;r--,s--
n = b[s] - 48; // n=5 n为B2的值
t = m * n + j + result[r]; // t=2*5+0+result[3]=10+0=10 将两个数相乘再加上原来这个位置的值
result[r] = t % 10; // result[3]=0 模10得到这个位置上的值,
j = t / 10; // j=1 r=2 s=0 整除10看是否需要进一位,如果需要进一位,则j+result[2]得到最终的值
}
// r=2,s=-1
for( ; j ; r -- ){
t = j + result[r]; // t=1+result[2]=1+3=4
result[r] = t % 10; // result[2]=4
j = t / 10; // j=0 r=1
}
}
}
总结:
get(a,b,0,0,0,0,1,2) 1x4 00 result[1]=4(初值)
get(a,b,0,0,1,1,0,1) 1x5 0 result[2]=5(初值)
get(a,b,1,1,0,0,0,1) 2x4 0 result[2]=3(初值) result[1]=5 (终值)
get(a,b,1,1,1,1,0,0) 2x5 result[3]=0(终值) result[2]=4(终值)
c语言:分治算法之大数相乘的更多相关文章
- 大数相乘算法C++版
#include <iostream> #include <cstring> using namespace std; #define null 0 #define MAXN ...
- 大数相乘----C语言
/* 大数相乘: 因为是大数,乘积肯定超出了能定义的范围,因此考虑用数组存储,定义三个数组,分别存储乘数,被乘数和积. 规则与平常手算一样,从个位开始分别与被乘数的每一位相乘,但是有一点不同的是:我们 ...
- 分治算法求乘方a^b 取余p(divide and conquer)
传统的计算方法为循环n个a相乘.时间复杂度为O(n). 如用分治算法,效率可提升至O(lgn). 结合recursive有 double pow(int a, int n){ ) ; ) return ...
- Karatsuba乘法--实现大数相乘
Karatsuba乘法 Karatsuba乘法是一种快速乘法.此算法在1960年由Anatolii Alexeevitch Karatsuba 提出,并于1962年得以发表.此算法主要用于两个大数相乘 ...
- leetcode 43 Multiply Strings 大数相乘
感觉是大数相乘算法里面最能够描述.模拟演算过程的思路 class Solution { public String multiply(String num1, String num2) { if(nu ...
- POJ 2389 Bull Math(水~Java -大数相乘)
题目链接:http://poj.org/problem?id=2389 题目大意: 大数相乘. 解题思路: java BigInteger类解决 o.0 AC Code: import java.ma ...
- 从两个平方算法到分治算法-java
先来看看问题的来源,假设有这么一个数组: 1 2 -5 4 -2 3 -3 4 -15 我们要求出其中连续字数组的和的最大值 例如这么可以很明显看出 4+ –2 + 3 + –3 + 4 = 6 所有 ...
- 计算几何 平面最近点对 nlogn分治算法 求平面中距离最近的两点
平面最近点对,即平面中距离最近的两点 分治算法: int SOLVE(int left,int right)//求解点集中区间[left,right]中的最近点对 { double ans; //an ...
- java版大数相乘
在搞ACM的时候遇到大数相乘的问题,在网上找了一下,看到了一个c++版本的 http://blog.csdn.net/jianzhibeihang/article/details/4948267 用j ...
随机推荐
- 20.LVM
1.在硬盘分好区或者部署为RAID 磁盘阵列之后,再想修改硬盘分区大小就不容易了.换句话说,当用户想要随着实际需求的变化调整硬盘分区的大小时,会受到硬盘"灵活性"的限制. 这时就需 ...
- Linux常用习惯和技巧
1.如果有些命令在执行时不断地在屏幕上输出信息,影响到后续命令的输入,则可以在执行命令时在末尾添加上一个&符号,这样命令将进入系统后台来执行.
- 一文弄懂-Netty核心功能及线程模型
目录 一. Netty是什么? 二. Netty 的使用场景 三. Netty通讯示例 1. Netty的maven依赖 2. 服务端代码 3. 客户端代码 四. Netty线程模型 五. Netty ...
- G - 棋盘游戏
小希和Gardon在玩一个游戏:对一个N*M的棋盘,在格子里放尽量多的一些国际象棋里面的"车",并且使得他们不能互相攻击,这当然很简单,但是Gardon限制了只有某些格子才可以放, ...
- Codeforces Round #627 (Div. 3) E - Sleeping Schedule(递推)
题意: 每天有 h 小时,有一序列 an,每次可以选择 ai 或 ai - 1 小时后睡觉,问从 0 次 0 时开始,最多在 l ~ r 时间段入睡多少次. 思路: 如果此时可达,计算此时可达的时间点 ...
- P3355 骑士共存问题 (最小割)
题意:nxn的棋盘 有m个坏点 求能在棋盘上放多少个马不会互相攻击 题解:这个题仔细想想居然和方格取数是一样的!!! 每个马他能攻击到的地方的坐标 (x+y)奇偶性不一样 于是就黑白染色 s-> ...
- Successor HDU - 4366 分块
代码+注释: 1 /* 2 题意: 3 一共有n个人,其中0号是总裁(金字塔顶尖).后面输入其他n-1个人的信息啊a.b.c,分别代表第i个人的上级是a,他的 4 忠诚度为b,他的能力为c.后面有m次 ...
- 同时拿到BATJMD的Offer是怎样的一种体验?
写在前面 又到了收割Offer的季节,你准备好了吗?曾经的我,横扫各个大厂的Offer.还是那句话:进大厂临时抱佛脚是肯定不行的,一定要注重平时的总结和积累,多思考,多积累,多总结,多复盘,将工作经历 ...
- C#枚举(一)使用总结以及扩展类分享
0.介绍 枚举是一组命名常量,其基础类型为任意整型. 如果没有显式声明基础类型, 则为Int32 在实际开发过程中,枚举的使用可以让代码更加清晰且优雅. 最近在对枚举的使用进行了一些总结与整理,也发现 ...
- NLP论文阅读一:Paper阅读方法
参考:https://pan.baidu.com/s/1MfcmXKopna3aLZHkD3iL3w 一.为什么要读论文? 基础技术:读论文中的related works可以帮助了解该领域的一些主要的 ...