首页 > 其他分享 >快速幂学习笔记

快速幂学习笔记

时间:2022-08-26 02:03:49浏览次数:54  
标签:return int LL long 学习 base 笔记 ans 快速

前言

快速幂很有用哦!!

目前本文还没有例题,因为没有什么好题啊。

以后看一下能不能找一些题目。

什么是快速幂

幂,也就是次幂,可以理解为计算 \(x^y\)。

由于 \(x^y\) 会特别大,所以一般都是求 \(x^y \bmod p\)。

朴素的做法如下:

#define LL long long
LL slow_pow(int x, int y, int p)
{
    //由于精度问题,开 long long 明显更保险。
    LL mul = 1;
    for (int i = 1; i <= y; i++) mul = (mul * x) % p;
    return mul;
}

这样做,时间复杂度是 \(O(y)\),太慢了。

快速幂可以巧妙地在 \(O(\log y)\) 的时间内计算幂。

哪怕 \(y\) 处于 long long 级别(\(y \le 10^{18}\)),计算速度都为常数。

快速幂实现方式有很多,下面讲解常见的两种实现方式:二进制递归分治

实现方法

方法 01

这个办法围绕一条明显成立的等式展开:\(x^{a+b} = x^a \cdot x^b\)。

实际上,如果 \(y = 2^n\) (\(n\) 是自然数),计算 \(x^y \bmod p\) 的时间复杂度就是 \(O(\log y)\)。

因为可以利用倍增的思想,先算出 \(x^1\),然后得到 \(x^2 = x^1 \cdot x^1\),进而得到 \(x^4 = x^2 \cdot x^2\),以此类推。

那么,我们可不可以将 \(y\) 分解成若干个 \(2^n\) 姓形式的数呢?

当然可以!二进制拆分!


举个例子,\(x = 2, y = 10\)。

我们把 \(y\) 进行二进制拆分,这里 \((10)_{10} = (1010)_2\)。

所以 \(x^{10} = x^2 \cdot x^8\)。

我们可以用变量 \(\texttt{base}\) 记录当前 \(x^{\texttt{二次幂}}\) 的值,按照上述的倍增思想,每次自乘即可。

一边计算二进制一边计算答案,每逢二进制的那一位的值为 \(1\),\(\texttt{ans}\) 乘上 \(\texttt{base}\)。

大致原理就是这样,直接给出 C++ 代码。

#define LL long long
int fast_pow(int x, int y, int p) //求 x^y mod p
//也有以 ksm 命名的,即快速幂的首字母。
{
	LL ans = 1, base = x;
    while (y) //等同于: while(y != 0)
    {
        //此处 y&1 是位运算常数优化,等同于 if (y % 2 == 1)
        if (y & 1) ans = (ans * base) % p;
        base = (bse * base) % p; //自乘,与上述原理相同。
        y >>= 1; //同样是位运算常数优化,等同于 y /= 2;
	}
    return (int)(ans);
}

挺好理解的吧,给出精简一些的代码,考试时就打下面这份,速度快一些。

#define LL long long
int ksm(int x, int y, int p){
    LL a=1, t=x;
    while(y){
        if(y&1) a=(a*t)%p;
        t=(t*t)%p;
        y>>=1;
	}
    return (int)a;
}

接下来,再看第二种更好理解的方法:递归。

方法 02

分两种情况考虑。

若 \(y\) 是偶数,\(x^y = x^{y \div2} \times x^{y \div2}\);

若 \(y\) 是奇数,\(x^y = x^{y \div2} \times x^{y \div2} \color{red}{\space \times\space x}\)。

利用递归求出 \(x^{y \div 2}\),然后按照两种情况计算出答案就完事了。

终止条件是 \(y = 1\),答案就是 \(x \bmod p\)。

代码如下:

#define LL long long
int fast_pow(int x, int y, int p)
{
    if (y == 1) return x % p; //终止条件 
    //下面 y >> 1 是位运算,等同于:y / 2 
    LL ans = fast_pow(x, y >> 1, p);
    if (y % 2 == 0) //与方法01相同,可以用位运算压,这里没有使用。
    {
    	ans = (ans * ans) % p;
    	return (int)(ans);
	}
	else
	{
		ans = (ans * ans) % p;
		ans = (ans * x) % p; //这两句话等同于 ans = ans * ans * x 这样乘一次模一次,防止数据溢出。 
		return int(ans);
	}
}

注意到两种情况都包含了语句:

ans = (ans * ans) % p;

以及这条语句:

return (int)(ans);

所以,我们可以简单地缩短码量,最终精简代码如下:

#define LL long long
int ksm(int x, int y, int p)
{
    if(y==1) return x%p;
    LL t=ksm(x, y>>1, p);
    t=(t*t)%p;
    if(y&1)t=(t*x)%p;
    return (int)t;
}

首发:2022-05-29 15:46:56

标签:return,int,LL,long,学习,base,笔记,ans,快速
From: https://www.cnblogs.com/liangbowen/p/16622880.html

相关文章

  • 状态压缩 DP 学习笔记【入门篇】
    前言状态压缩DP,简称状压DP。之前一直觉得状压特别难,学了一下,发现基本形态挺简单的。在学习之前,你需要掌握:简单DP(如线性DP,背包)基本二进制运算:&运算、|运算、\(......
  • 拓扑排序学习笔记
    前言在学习这篇文章之前,你需要了解的算法有:基本图论知识链式前向星(图的一种存储方式)了解队列、栈等简单数据结构的实现,用STL也行。什么是拓扑排序AOV网的定义......
  • java学习:八大基本类型变量
    1.类在java中用class来定义一个类,类是java程序的基本单位类描述的是具有共性的一类事物,所以我们又可以把类称作为模板技术 如何理解共性:具有相同的属性--》j......
  • 05.爬虫入门笔记1
    入门爬虫笔记011.request库的使用使用request库的get方法importrequestr=request.get('www.baidu.com')这会得到一个Response对象,将其存入变量r。显示得到的......
  • SSCMS文件解析-学习笔记
    //声明常量,不可变constfs=require('fs-extra');//初始化目录插件constdel=require('del');//删除文件的工具constgulp=require('gulp');//基于流的代码自动化......
  • python学习Day53
    Day53今日内容概要JS数据类型JS数据类型—布尔值JS数据类型—对象objectJS数据类型—自定义对象objectJS运算符JS流程控制JS函数JS内置对象JS的BOM与DOM操......
  • 学习python-Day47
    今日学习内容JS数据类型比较我们学过python的数据类型去学习布尔值python:boolTrue:数字False:0None''[]{}...JS:boolentruefa......
  • 电子笔记本的思考(1)(ver0.4)
    章节: 电子笔记本的思考(1)(ver0.4)上面的是截屏的完整版,分割线下面的是纯文字版本: 作者姓名(本人的真实姓名):胡佳吉居住地:上海作者网名:EverSteins版权声明:电子笔记本的......
  • JavaScript快速入门-06-函数
    6函数6.1函数定义  函数可以封装语句,然后在任何地方、任何时间执行。JavaScript中的函数使用function关键字声明,主要由函数名、函数参数和函数体组成。其基本语法......
  • 前端学习-4
    目录JS数据类型之布尔值JS数据类型之对象(object)JS数据类型之自定义对象(object)运算符流程控制函数JS内置对象BOM与DOM操作8.25小练习JS数据类型之布尔值回顾一波pyth......