• 快速乘详解


    两个long long 的数相乘,模一个long long的数,怎么处理?

    直接乘是肯定不行的,于是我们就要用到快速乘。

    1.用long double

    首先,我们知道 a % b = a − ⌊ a b ⌋ × b a\%b=a-\lfloor\frac{a}{b}\rfloor\times b a%b=aba×b,所以 x × y % p = x × y − ⌊ x × y p ⌋ × p x\times y\%p=x\times y-\lfloor\frac{x\times y}{p}\rfloor\times p x×y%p=x×ypx×y×p

    这里 x , y , p x,y,p x,y,p都是long long的,显然 x × y x\times y x×y可能会溢出,但是减去 ⌊ x × y p ⌋ × p \lfloor\frac{x\times y}{p}\rfloor\times p px×y×p之后的结果肯定是小于 p p p的。溢出后减回来结果是一样的。时间复杂度为 O ( 1 ) O(1) O(1),适合用于卡常数的题目。

    但是,因为long double有精度问题,所以如果数组过大的话返回值就会出问题。

    code

    long long ch(long long x,long long y,long long p){
    	return (x*y-(long long)((long double)x/p*y)*p+p)%p;
    }
    
    • 1
    • 2
    • 3

    2.用二进制

    快速幂原理相似,原本 x × y x\times y x×y x + x + x + ⋯ + x x+x+x+\dots+x x+x+x++x连续加 y y y x x x。我们可以优化一下,对于 y y y转为二进制中的第 i i i位,如果这一位为1,则 a n s + = x × 2 i ans+=x\times 2^i ans+=x×2i。所以只需枚举 i i i,具体与快速幂相似,时间复杂度为 O ( y ) O(y) O(y)

    code

    long long ch(long long x,long long y,long long p){
    	long long re=0;
    	for(;y;y>>=1,x=(x+x)%p){
    		if(y&1) re=(re+x)%p;
    	}
    	return re;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
  • 相关阅读:
    断点是什么,断点有哪几种类型?
    直接插入排序
    写作技巧网站或工具
    *javaSE 面试题
    web缓存器和CDN
    C#面:请解释ASP.NET中的web页面与其隐藏类之间的关系
    写给Python社群的第8课:Python异常,你必须掌握的技术点
    Go 单元测试之mock接口测试
    maven 安装本地jar失败 错误指南
    【docker】数据卷和数据卷容器
  • 原文地址:https://blog.csdn.net/tanjunming2020/article/details/126901527