当前位置:首页 > C++知识 > 正文内容

【数论】龟速乘

亿万年的星光2年前 (2023-02-04)C++知识7603

我们前面一篇文章学习了快速幂。它可以解决两类问题:

  • a^b,

  • (a^b)%c

对于第一类,我们可以使用递归法或者迭代法可以求出,为了计算的快,我们可以引入位运算操作,但是目前来看,无论怎么优化都不能超过long long。

对于第二类,是快速幂的优点所在,通过 (a*b)%m=(a%m * b%m) %m公式,我们可以将结果运算范围大大减小,使运算结果保持在m范围以内。

但是,快速幂并不是万能的,比如下面这个例子求(a^b)%m。

19260817 2333333 1234567654321

也就是(19260817^233333)%1234567654321。那么很明显,快速幂也会爆掉。因为取模的数> 1e9后,两个数相乘就会爆 long long了(高精度除外)。


那么这样的题目应该怎么处理,这就是我们今天的主角:龟速乘


一、龟速乘

简单来说,当形如 (a*b)%c这种表达式进行求解计算时,如果a,b的值都>=1e9,特别当c的值也>=1e9时,long long 会爆掉,此时采用龟速乘的方式来进行计算。

顾名思义,龟速乘是比较慢的,主打的就是 慢工出细活

代码模板如下:

long long slowmul(long long x,long long y,long long mod) {
	long long ans=0;
	while(y) {
		if(y&1==1){
		 ans+=x;
		 ans%=mod;
		}
		x=x+x;
		x%=mod;
		y>>=1;
	}
	return ans;
}


二、龟速乘原理

要知道龟速乘的原理,首先要回到小学阶段一开始讲乘法原理的时候:乘法的本质就是加法

比如下面这个例子:

我们计算3*5。

3*5 = 3*(4+1)   //这个地方为什么是4+1,而不是2+3?
    = 3*4 + 3*1

//看到这里,有没有什么想法
//如果没有,我们借助一个图来清楚描述一下刚才的过程
7*13=3*(8+4+1)


上面的这个过程,1 1 0 1 对应 8 4 2 1。非常巧合的一件事,8 4 2 1 就是2^3、2^2、2^1、2^0。很显然,我们回到了二进制,而龟速乘就是利用了这个原理来进行操作的。

在上面的过程中,8 4 2 1是二进制中基,从左往右是除2,从右往左是乘2,对应的二进制操作中右移和左移。

那么我们再来详细看看代码是怎么处理的。

我们用这种方法求7*13:

int slowMult(int a,int b){ //a为基底,b为乘数,即a*b
	int ans=0; //ans即最终结果
	while(b) {
		if(b&1) 
			ans+=a; // 判断二进值是否为1
		a+=a; //基底乘2
		b>>=1;//位运算
	}
	return ans;
}

注意,此方法区别于我们前面讲的迭代法求快速幂。

//迭代法求快速幂,求a^b
#include<bits/stdc++.h>
using namespace std;
long long pow_d(long  a,long  b){
    long ans=1;
    while(b>0){
        if(b&1){//如果b的二进制末尾为1 ,就相当于被选中了。
        //就如2^13 ==> 2^(13==>1101)==> 2^(1101) ==> 3 2 0 号为 1 那么被选中 ==> 2^13 = 2^8 *  2^4 * 2^1
            ans=ans*a;//令ans累积上a 
        }
        a=a*a;//令a平方 
        b>>=1;//将b的二进制右移一位即 
    }
    return ans;
}
int main(){
    long long  a,b;
    cin>>a>>b; 
    long long  result=pow_d(a,b);
    cout<<result<<endl;

区别:快速幂里的x是指数级增长,而龟速乘变成了翻倍。


我们再回来看(a*b)%c

long long slowmul(long long x,long long y,long long mod) {
	long long ans=0;
	while(y) {
		if(y&1==1){
		 ans+=x;
		 ans%=mod;
		}
		x=x+x;
		x%=mod;
		y>>=1;
	}
	return ans;
}


扫描二维码推送至手机访问。

版权声明:本文由青少年编程知识记录发布,如需转载请注明出处。

分享给朋友:

相关文章

2023 CSP 山东地区分数线汇总

地区CSP-XCSP-JCSP-S烟台556648.5临沂516416青岛476753淄博446547.5...

如何判断回文数/回文串

所谓回文,就是从左往右读和从右往左读都是一样的,这样的数字或者字符称为回文数/回文字符。做题的时候经常能看到判断回文操作。判断回文的一般有两种,一种是数字类型,一种是字符类型。两种分别介绍一下。一、回...

C++将数据写入磁盘文件

0.前言要求:在任意路径下新建一个文本文档,向该文档中写入数据。以'#'结束字符串的输入。关键技术:ch=fputc(ch,fp);该函数的作用是把一个字符写到磁盘文件(fp所指的磁盘...

【题解】围圈报数(约瑟夫问题)

【题解】围圈报数(约瑟夫问题)

【题目描述】有n个人依次围成一圈,从第1个人开始报数,数到第m个人出列,然后从出列的下一个人开始报数,数到第m个热呢又出列,... ,如此反复到所有的人全部出列为止。设n个人的编号分别为1,2,......

STL入门——容器3:map

一、定义    Map是STL的一个关联容器,它提供一对一(其中第一个可以称为关键字,每个关键字只能在map中出现一次,第二个可能称为该关键字的值)的数据&nb...

【数据结构】栈—表达式括号匹配

【数据结构】栈—表达式括号匹配

【题目描述】假设一个表达式有英文字母(小写)、运算符(+,—,*,/)和左右小(圆)括号构成,以“@”作为表达式的结束符。请编写一个程序检查表达式中的左右圆括号是否匹配,若匹配,则返回“YES”;否则...