算法之高精度(七)

前一篇内容简单描述了:高精度数字和常规数字相乘
相对于乘法,高精度除法在高精度算法中比较难的部分。原因是不论加减乘的内容,我们始终只需要去计算,而除法就会涉及到判断。我们必须了解到被除数÷除数=商…余数余数不能大于除数,也不能小于0。这是我们需要控制的。
这篇我们将详细介绍两个高精度数字相除。首先,我想我还是需要提醒一下大家。

除法的原理是什么?

我常常称除法是减法的快速运算。如何理解这句话呢?我们设想,一个盘子里有10个苹果,需要分给3位小朋友,请问每位小朋友能分到几个?你会很快反应出来,3个。你的结果基于除法,10÷3=3...1。结果会剩下一个苹果。在学习除法之前,当你被问及这个问题时,我们常常会如此解释,每位同学每人拿一个,10-3=7,剩下7个,显然每位同学还可以拿。7-3=4,剩下4个,很显然还是可以拿。4-3=1,结果为1个,不够每位同学拿一个了。到这里就计算结束了。每位同学拿的次数就是分到的苹果数。剩下的苹果数量就是余数。
按照这样的道理,高精度数字相除也可以使用减法模拟出来,只不过需要一直使用被除数减去除数,直到余数小于除数。这是一个好方法。但对于两个高精度数字相差太大的话,减去的次数就有些不可控了。例如:346,334,834,985,792÷32,782,321,649——15位数除以11位数,使用减法模拟除法的方式,我们需要减10000多次,甚至不止如此。这是非常庞大的计算量。我有时会很担心使用这种方法计算,我的电脑能不能坚持运行完这个程序。
我们尝试使用一个新的方法。我们在数学书里很少简单,但生活中却运用自如的方式。我们假设1600÷13,你会怎么想?1600会分解成1300+300,其中1300÷13=100,剩下的300才继续运算。使用这种方法计算便会简单许多,但这个方法可能会因数字而异,对于一个难拆分的数字,我们就会犯难了。比如说12340÷234,你会想到怎么拆分?
我们采用补0的方式会更好。我们使234变成2340,然后在将12340÷2340 = 5,计算速度快了许多,5将作为十位数,剩下的余数再次对234求解。当然,这里的除法我们仍然使用减法进行模拟,也就是说,使12340减去2340,减去5次,就可以得到余数小于除数的情况。本来应该减去50次的量,转变成了5次,这样的改变我们将为之欢喜。
文字的描述总没有表格直观,我们使用表格将文字内容转换出来。这样更简洁。

下标01234
被除数04321
第一次转换的除数00432
第二次转换的除数04320
第三次转换的除数43200

表格中被除数减去第一次转换的除数,得到结果作为百位数字,结果为0,减去之后结果为余数。余数减去第二次转换的除数,得到结果作为十位数字,结果为5,减去之后的结果为余数。再次使用新产生的余数减去第三次转换的除数,得到结果作为个位数字。这里的减去指的是减到余数小于除数
这是我们用作两个高精度数字相除的方式。下面我们将尝试使用程序描述这一过程。

//假设a1,a2为两个高精度数字,a1为被除数,a2为除数
//假设a1,a2都已定义,且逆向存放
//length1, length2 分别表示a1,a2的位数

#include <iostream>

using namespace std;

int a1[100] = {};
int a2[100] = {};

//判断余数是否小于除数
//false 表示余数大于等于除数,此时应继续使用减法
//true 表示余数小于除数,不应继续使用减法
bool isSmall(int a1[], int tmp[], int length1, int length2)
{
	if(length1 > length2)	//余数位数大于除数位数
		return false;
	if(length1 < length2)	//余数位数小于除数位数
		return true;
	//两个位数相同,从高到低比较每一位,某一位越大,数字越大
	for(int i = length1 - 1; i >= 0; --i)
	{
		if(a1[i] > tmp[i])	//余数数字大于除数数字
			return false;
		if(a1[i] < tmp[i])	//余数数字小于除数数字
			return true;
	}
	return false;	//余数与除数完全相同,应继续使用减法
}

//减法,被除数减去除数得到余数
//计算结果存放在a1中,由于a1的位数可能发生变化,因此使用 &length 引用传参
void sub(int a1[], int tmp[], int &length1, int length2)
{
	for(int i = 0;i < length1; ++i)
	{
		a1[i] -= tmp[i];	//对应数位相减
		if(a1[i] < 0)	//结果可能需要借位
		{
			a1[i + 1] -= 1;	//借位,高位减少1
			a1[i] += 10;	//本位转换为正数
		}
	}

	//计算最终a1中有效位数
	//去前导0
	int k = length1 - 1;
	while(a1[k] == 0 && k >= 0)
		--k;
	//经过循环得到的k为高位第一个不为0的数字对应的下标,并不是位数
	length1 = k + 1;	//将结果位数调整至实际情况
}

//两个高精度整数相除
void division(int a1[], int a2[], int length1, int length2)
{
	int result[100] = {};	//假设a1,a2的商不超过100位数,result用来存放商
	//商的位数,商的位数不超过被除数的位数减去除数的位数 + 1
	int length = length1;	
	for(int i = length; i >= length2; --i)
	{//i表示每次转换时的总位数
		int tmp[100] = {};	//临时保存转换后的除数
		//转换除数
		for(int j = 0; j < length2; ++j)
			tmp[j + i - length2] = a2[j];
		while(isSmall(a1, tmp, length1, i) == 0)
		{
			sub(a1, tmp, length1, i);
			++result[i];
		}
	}//经过多轮计算,得到的商存放在result数组中,并逆向存放
	//余数存放在a1中
	int k = length;
	while(result[k] == 0 && k > length2)
		--k;
	for(int i = k; i >= length2; --i)
		cout << result[i];		//输出商
	cout << endl;
	
	//输出余数
    if(length1-1 < 0)
        cout << 0 << endl;
    else
        for(int i = length1 - 1; i >= 0; --i)
            cout << a1[i];	//输出余数
}

int main()
{
    string s1;
    string s2;
    cin >> s1 >> s2;

    int len1 = s1.size();
    int len2 = s2.size();
    for(int i = 0; i < len1;++i)
        a1[i] = s1[len1-i-1] - '0';
    for(int i = 0; i < len2; ++i)
        a2[i] = s2[len2-i-1] - '0';

    division(a1,a2,len1, len2);
}

按照这样的方式,两个高精度数字相除的程序我们总算描述结束了。其中还是有一些需要提醒的地方。例如,参数中使用了引用传参的方式,使用多个函数辅助等等。
我觉得,对于上述描述的求解方式应该是大家关注的点。至于代码描述,每一个人都可能实现出不同的版本。当然,代码中每一处细节,我还是希望各位能够钻研。这是有益的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值