大数(加、减、乘、除、低精度*大数)模板详解(C++)

一、大数四个基本操作

首先,我们来了解什么是大数,大数就是指用我们平常常见的高级语言(如:c 、c++)的基本数据类型的最大长度都装不下的数据,例如(1234567899876543211234567896542132165465),这些只能用字符数组(char[])或者字符串(string)来处理,大数操作最基础的四个操作就是:大数加法、大数减法、大数乘法、大数除法。下面我来详细介绍下四个基础操作的算法。

 

在介绍算法之前,先了解一下几个C++中 string数据类型的用法。因为string类字符元素的访问比C字符串有所增强,优点十分多,用起来很方便,所以下面我大数模板全部是基于string来作为基础数据类型。

String 优点如下(相对C

1string可以像C的字符串数组一样用下标来直接访问索引值对于的字符。

string str=”123456789123123”  //初始化一个字符串

str[i]                 //返回str中索引i处字符的引用,不查是否出界

str.at(i)              //返回str中索引i处字符的引用,查是否出界

2string重载了一些运算符,特别注意当目标串较小,无法容纳新的字符串,系统会自动分配更多的空间给目标串,不必顾虑出界:

 

3string字符串之间的赋值不再需要像C那样移动数组或者用strcpy()等函数辅助,string类可以直接赋值:

str1=str2;                     //str1成为str2的代码

str1+=str2;                    //str2的字符数据连接到str1的尾部

str1+str2;                 //返回一个字符串,它将str2和str1连接起来

4string字符串之间的比较也不需要用strcmp()等函数来比较,可以直接用<>==

str1==str2; str1!=str2;       //比较串是否相等,返回布尔值

str1<str2;  str1>str2;         //基于字典序的比较,返回布尔值

str1<=str2;  str1>=str2;

5) string字符串还有一个标准且强大的STL库,里面提供了许多非常便利有用的辅助函数,如 str. erase ()函数(详细介绍这个,因为下面算法经常利用到)

erase函数的原型如下:
(1)string& erase ( size_tpos = 0, size_t n = npos );(常用)

     作用:从下标size_t pos开始,去掉size_t n个字符

     例子:str=”023543”

     用法:str.erase(0,1);

     结果:str=“23543”
(2)iterator erase ( iteratorposition );

     作用:去掉一个下标为:position的字符

     例子:str=”023543”

     用法:str.erase(0);

     结果:str=“23543”
(3)iterator erase ( iteratorfirst, iterator last );

作用:删除从first到last之间的字符(first和last都是迭代器)

好了,了解string到这里也差不多了。接下来就是算法分析

一、大数加法

原理:首先我们要对字符串进行初始化处理,就是给它们前面补上一个‘0’,防止有进位,如果没有进位我们最后可以用erase()函数将其删掉,然后直接对字符串进行操作,从字符串尾开始操作,往回对应相加,如果相加的结果>10,那么前一个字符就要加上当前字符的进位,当前字符就只保留结果个位数

下面是图解:


代码如下

二、大数减法

原理:减法操作和加法操作类似,首先先初始化,保持输入都是大数减小数,然后从低位开始对应相减,不够减就从借位减一。

图解:


代码如下


三、大数乘法

原理:乘法的主要思想是把乘法转化为加法进行运算。

例子(12345*24)首先用相对小的大数(24)作为循环分割,4和 20,个位数4代表着4个123456相加。

        12345*4=12345+12345+12345+12345

20代表着20个12345相加,也就是2个123450相加

        12345*20=123450+123450

最终结果就是4个12345加上2个123450

        12345*24=12345+12345+12345+12345+123450+123450

代码如下:


                                                                                        不明白?

                                                                                     


我带大家走一遍Debug,看一下数据的变化

首先初始化:


循环加上4次加上12345后结果res=49380


重复,循环2次加上123450,res=296280


最终结果


四、大数除法

原理:首先想到的是运用大数减法,例如144 / 12,就是循环利用大数减法144-12,减一次就用大数加法加1,最终结束循环的条件是被减数小于减数。这个方法实现起来简单,逻辑直观,但是效率明显不够高,不信可以试一试(978973548932486564654654654654654654654564564564654532165454654 /2)

这个运用上面的方法会编译器会直接卡死。其实呢,上面的只要进行优化一下就可以AC了。怎么优化呢?

这样,首先我们可以直接将减数扩展到刚刚好比被减数少一位(用‘0’扩展,就是乘于10,例如这里刚好是扩大10倍)然后再大数相减,减一次就用大数加法加起来(这里是加10),循环操作,同样,循环结束条件也是被减数小于减数,这样可以避免循环减一个小数的尴尬。

代码:


                                                                                        不明白?

                                                                                     

下面走一遍debug,看一下数据变化

测试样例(9090901 / 9

第一步:扩展减数,在这里减数从9扩展为900000,相当于*100000

第二步:接着循环相减直至被减数小于扩展的减数即(a<tmp)结束,此时也循环大数相加加上了对应的扩展倍数就是结果(res)

由于被减数(a=90901)还是大于减数(b=9),循环继续

此时被减数(a=90901)小于扩展的除数(tmp=900000)了,需要将扩展的除数变小(这里将它缩小10倍,即tmp=90000)图如下:

然后重复第二步的操作,直至被减数小于减数即(a<b)结束。

最终结果:

二、低精度*大数:

     低精度*大数例如阶乘(N!),1000!将会是一个非常大的数,用我们的基本数据类型是无法装下的,1*2*3*4*·······*10000

当然这利用我们前一小节所说的大数乘法完成该功能也可以,但是还有一种专门用来处理这种情况的方法,就是利用一个数组来保存每一位的字符。

例如:

5的阶乘在数组里面就是:

3

0

2

1

 0

 0

 0

 0

 0

 0

6的阶乘在数组里面就是:

3

0

2

7

 0

 0

 0

 0

 0

 0

7的阶乘在数组里面就是:

4

0

4

0

5

 0

 0

 0

 0

发现吗?

数组的第一个元素其实就是表示当前结果的位数,其他元素就是结果的每一位数

5!=120

6!=720

7!=5040

那其中是如何实现的呢

代码如下:


首先使用一个数组num[50000]来存储结果,将其初始化为0num[0]=1num[1]=1

这里设定num[1]=1是因为阶乘的原因,因为0的阶乘还是1

文字表达水平有限,原理还是看我debug走一遍更好理解

 

测试样例是6的阶乘(测试样例太大不好讲解)

做好初始化工作:i=1;                                 [结果]res=1


第一步判断num[0]此时的结果有多少位,再循环分别每位乘以i (第一轮i=1)

循环后数组没有变化

1

1

0

0

 0

 0

 0

 0

 0

 0

第二轮循环开始:i此时为2 res*2 [注意,这里*2是指结果的每一位乘以2] res=2


数组变化

1

2

0

0

 0

 0

 0

 0

 0

 0


第三轮循环开始:i=3,res*3

数组变化

1

6

0

0

 0

 0

 0

 0

 0

 0


第四轮循环开始: i=4res*4



注意这里出现了24,没错,就是6*4=2424>10了,此时就要进位,向后进位,原位置保持个位数,num[0]就要加多一位了,代表这结果的位数

数组变化

2

4

2

0

 0

 0

 0

 0

 0

 0

第五轮循环开始: i=5res*5

每位分别乘以5,4*5=20  2*5=10

再循环分割进位

数组变化

3

0

2

1

 0

 0

 0

 0

 0

 0

第六轮循环开始: i=6res*6

同样,每一位都分别乘以6

即:0*6=0 2*6 =12 1*6=6

扫描有大于10的进行进位拆分


数组变化

3

0

2

7

 0

 0

 0

 0

 0

 0

结束循环

最终:模板代码如下

#include<cstdio>
#include<iostream>
#include<cstring>

using namespace std;
//初始化 
void initial(string &a, string &b){
	while (a.size()<b.size())a = '0' + a;
	while (b.size()<a.size())b = '0' + b;
}
//打印 
void print(string &a, string &b){
	cout << a << endl;
	cout << b << endl;
}
//找出最大的字符串 
void findMax(string &a, string &b){
	string tmp;
	if (a<b){
		tmp = b;
		b = a;
		a = tmp;
	}
}
//删除第一个字符'0' 
bool del(string &a){
	if (a[0] == '0'){
		a.erase(0, 1);
		return true;
	}
	else
		return false;
}
//删除前面所有的 0 
void delAllZroe(string &a){
	while (del(a)){
		del(a);
	};
}
//大数加法 
string bigItergeAdd(string a, string b){
	initial(a, b);
	a = '0' + a;
	b = '0' + b;
	for (int i = a.size() - 1; i >= 0; i--){
		int num1 = a[i] - '0';
		int num2 = b[i] - '0';
		if (num1 + num2>9){
			a[i - 1] = a[i - 1] - '0' + 1 + '0';
			a[i] = (num1 + num2) - 10 + '0';
		}
		else{
			a[i] = (num1 + num2) + '0';
		}
	}
	del(a);
	//	cout<<a<<endl;
	return a;
}
//大数减法 
string bigItergeSub(string a, string b){
	initial(a, b);
	findMax(a, b);
	for (int i = a.size() - 1; i >= 0; i--){
		int num1 = a[i] - '0';
		int num2 = b[i] - '0';
		if (num1<num2){
			a[i - 1] = a[i - 1] - '0' - 1 + '0';
			a[i] = (num1 + 10 - num2) + '0';
		}
		else{
			a[i] = (num1 - num2) + '0';
		}
	}
	del(a);
	//	cout<<a<<endl;
	return a;
}
//大数乘法(大数加法实现) 
void bigItergeMul(string a, string b){
	delAllZroe(a);
	delAllZroe(b);
	if (a == "" || b == ""){ printf("0\n"); return; }
	initial(a, b);
	findMax(a, b);
	string res = "0";
	int count = 0;
	delAllZroe(b);
	for (int i = b.size() - 1; i >= 0; i--){
		int num1 = b[i] - '0';
		if (i != b.size() - 1)		a = a + '0';
		for (int i = 1; i <= num1; i++){
			res = bigItergeAdd(res, a);
		}
	}
	delAllZroe(res);
	cout << res << endl;
}
//大数除法 
void bigItergeDiv(string a, string b){
	initial(a, b);
	if (a<b){ cout << "0" << endl;	return; }
	delAllZroe(b);
	string res = "0";
	string restmp = "1";
	string tmp = b;
	for (int i = 1; i<(a.size() - b.size()); i++){
		tmp += '0';
		restmp += '0';
	}
	initial(a, b);
	while (a >= b){
		initial(a, tmp);
		if (a >= tmp){
			a = bigItergeSub(a, tmp);
			res = bigItergeAdd(res, restmp);
		}
		else{
			tmp.erase(tmp.size() - 1);
			restmp.erase(restmp.size() - 1);
			initial(a, tmp);
			if (a >= tmp){
				a = bigItergeSub(a, tmp);
				res = bigItergeAdd(res, restmp);
			}
		}
		initial(a, b);
	}
	cout << res << endl;
}
//阶乘(0~10000)【实际是低精度 乘于(*) 大数 例如:1000 *32132156465465321】 
void factorial(int n){
	int num[50000];
	memset(num, 0, sizeof(num));
	num[0] = 1;
	num[1] = 1;
	for (int i = 1; i <= n; i++){
		int len = num[0];
		for (int j = 1; j <= len; j++){
			num[j] *= i;
		}
		for (int j = 1; j <= num[0]; j++){
			if (num[j]>9){
				num[j + 1] += num[j] / 10;
				num[j] %= 10;
			}
			if (num[num[0] + 1] != 0)num[0]++;
		}

	}
	for (int i = num[0]; i>0; i--){
		printf("%d", num[i]);
	}
	printf("\n");
}

int main(){
	string a, b;
	while (cin >> a >> b){
//		bigItergeAdd(a,b);
//		bigItergeSub(a,b);
		bigItergeMul(a,b);	
//		bigItergeDiv(a, b);
	}
//	int n;
//	while(scanf("%d",&n)!=EOF){
//		factorial(n);
//	}
	return 0;
}

以上的模板已经在acm.hdu.edu.cnopenjudge.cn上测试AC

大数加法

http://acm.hdu.edu.cn/showproblem.php?pid=1002

http://bailian.openjudge.cn/practice/1000/

http://acm.hdu.edu.cn/showproblem.php?pid=1047

http://acm.hdu.edu.cn/showproblem.php?pid=1313

 

这题在输入做了手脚,输入0,也要输出0WA了好几次

大数减法

http://bailian.openjudge.cn/practice/2736/

 

大数乘法

http://bailian.openjudge.cn/practice/2980/

这道题目特坑,竟然在输入做手脚,测试样例会输入一串00000000000000

或者 000000000001 * 000000000000001等等,在这上面WA了好几次

 

大数除法

http://bailian.openjudge.cn/practice/2737/

 

低精度*大数(N!)阶乘

http://acm.hdu.edu.cn/showproblem.php?pid=1042

 

这些都是一些非常基础的题目,一眼就看出是大数的运用,但实际运用上,大数只是作为解题的一个桥段,作为一个契机,例如前一篇文章

sicily1029Rabbit  中大OJ解题报告

http://blog.csdn.net/wjb820728252/article/details/60583288


  • 12
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
### 回答1: 在C语言中进行大数运算,一般有以下几种方式: 1. 自己实现高精度计算库:通过定义自己的数据结构(比如用数组表示大整数)和实现基本的大数乘除等运算,可以实现高精度计算。但是这需要自己实现大量的代码,而且效率可能不如专业的大数计算库。 2. 使用现有的大数计算库:有一些现成的开源大数计算库(比如GNU MP库),可以在C语言中直接调用。使用这些库可以避免自己实现大量的代码,同时也可以获得高效的运算性能。 无论采用哪种方式,都需要了解大数运算的原理和具体实现方式,以及注意处理边界情况和错误情况,以保证计算的正确性和可靠性。 ### 回答2: 大数运算是指处理超过计算机整数位数上限的数值运算。C语言是一种基础的编程语言,它没有内置针对大数运算的数据类型和运算符,但可以通过自定义数据结构和算法来实现大数运算。 一种常用的实现大数运算的方法是使用数组来表示大数。具体步骤如下: 1. 定义一个足够长的数组,用于存储大数的每一位,数组的长度应根据需要处理的数值范围确定。 2. 将大数按照逆序的方式存储在数组中,即个位数存储在数组的第一个元素,十位数存储在数组的第二个元素,依次类推。 3. 实现基本的加法减法乘法除法运算函数。对于加法减法,可以从个位数开始逐位相或相,将结果存储在新的数组中,并考虑进位或借位;对于乘法除法,可以参考手动计算的方法,将运算结果逐步存储在新的数组中。 4. 根据需要,还可以实现其他的运算操作,如取模运算、幂运算等。 需要注意的是,由于大数计算涉及到多位数的运算,所以处理起来相对较慢,需要考虑计算时间和内存占用。同时,大数运算中也需要考虑到运算结果溢出的问题,以及负数运算的处理等。 综上所述,通过使用数组和自定义算法,我们可以利用C语言进行大数运算。这样可以扩展C语言在数值计算方面的能力,应对一些特殊的计算需求。 ### 回答3: 在C语言中进行大数运算可以通过使用字符串来表示和计算大数。以下是一种简单的实现方法: 1. 首先,将要进行运算的大数转换成字符串表示。可以使用字符数组来表示,例如char number[100],其中100是一个足够大的长度以容纳大数。 2. 定义一个结构体来表示大数,结构体中包含一个字符数组和一个记录大数位数的变量。例如: ```c struct BigInt { char digit[100]; int length; }; ``` 3. 编写函数来实现大数的基本运算,例如加法减法乘法除法。对于加法减法运算,可以模仿手动计算时的竖向计算方法,从位数开始逐位相或相,同时注意进位或借位。 4. 对于乘法除法运算,可以借助于循环和进位(借位)的操作,逐位相乘或相除,并将结果保存在一个新的字符数组中。 5. 在进行运算时,为了方便操作,可以将字符串的字符顺序进行逆序,从而在计算过程中更容易按照从位到高位的顺序进行。 6. 编写其他必要的辅助函数,例如比较大小、取反等。 请注意,大数运算是一种复杂的问题,需要考虑到边界情况和错误处理。以上仅是一种简单的实现方法,对于更复杂的大数运算问题还需要进一步的优化和改进。

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

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值