updata on 2020.4.11

修正了 excrt 的一处笔误

CRT

求解方程:

\[\begin{cases}
x \equiv a_1 \pmod {m_1}\\
x \equiv a_2 \pmod {m_2}\\
\vdots \\
x \equiv a_n \pmod {m_n}\\
\end{cases}
\]

其中,保证\(m_i\)是两两互质的正整数,crt就是基于这个特征

我们记\(M=\prod_{i=1}^{n} m_i\),和\(M_i=\dfrac{M}{m_i}\),\(t_i\)满足\(M_it_i \equiv 1 \pmod {m_i}\)

即\(M_i\)是所有下标不为\(i\)的\(m\)乘起来,而\(t_i\)是\(M_i \bmod m_i\)的逆元

则对于任意的\(k\neq i\),\(a_iM_it_i\equiv 0\pmod {m_k}\),因为\(M_i\)中一定包含了\(m_k\)这个因数

而又因为\(M_it_i\equiv1\pmod{m_i}\),所以\(a_iM_it_i\equiv a_i\pmod{m_i}\)

所以可以说明

\[\sum_{i=1}^n a_iM_it_i
\]

为问题的一组解,且这个解在\(\mod M\)下唯一

当然通解就是\(x+kM,k\subset Z\)

当然求逆元的过程要用到exgcd,而题目要求最小正整数解,所以只要让解在\([0,M-1]\)中就行了

但是,如果直接\(M=\prod_{i=1}^n m_i\)会乘爆,要让\(M=\text{LCM}_{i=1}^n m_i\),可以有和上文叙述的一样的性质,下面的excrt也是一样

题目代码

#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline int read(){
int x=0,y=1;
char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
int n;
LL a[15],m[15],M=1;
void exgcd(LL a,LL b,LL &x,LL &y){
if(!b){
x=1;y=0;
return;
}
exgcd(b,a%b,x,y);
LL tmp=x;x=y;
y=tmp-a/b*y;
}
int main(){
n=read();
for(reg int i=1;i<=n;i++){
m[i]=read();a[i]=read();
M=M/std::__gcd(M,m[i])*m[i];
}
LL ans=0,Mi,x,y;
for(reg int i=1;i<=n;i++){
Mi=M/m[i];
exgcd(Mi,m[i],x,y);
ans=((ans+Mi*x*a[i])%M+M)%M;
}
std::printf("%lld",ans);
return 0;
}

EXCRT

还是求解方程:

\[\begin{cases}
x \equiv a_1 \pmod {m_1}\\
x \equiv a_2 \pmod {m_2}\\
\vdots \\
x \equiv a_n \pmod {m_n}\\
\end{cases}
\]

但这次不保证\(m_i\)两两互质

不过这好像和CRT关系不大

考虑用数学归纳法,假设我们已知前\(i-1\)的解为\(ans\),并记\(M=\text{LCM}_{k=1}^{i-1}m_k\) 则其通解为\(ans+M\times t\)

那么,我们就要确定一个\(t\),使得\(ans+M\times t\equiv a_i\pmod {m_i}\)

然后新的\(ans=ans+M\times t\)

考虑求解上面那个同余方程的方法

因为exgcd可以求解方程\(ax+by=\gcd(a,b)\)

那么我们转变那个同余方程的形式:

\[Mt\equiv a_i-ans \pmod{m_i}
\]

\[Mt+m_iy=a_i-ans
\]

那么如果\(\gcd(M,m_i)|(a_i-ans)\),则有解,题目保证了有解

用exgcd求出的是:

\[Mt'+m_iy=\gcd(M,m_i)
\]

那么让等式两边同时除以这个\(\gcd\)再同时乘以\(a_i-ans\)就行了

\[Mt'\dfrac{(a_i-ans)}{\gcd(M,m_i)}+m_iy\dfrac{(a_i-ans)}{\gcd(M,m_i)}=a_i-ans
\]

那我们要求的这个\(t\)就是\(t'\dfrac{(a_i-ans)}{\gcd(M,m_i)}\)

注意在乘的时候会爆long long,要用int128或是龟速乘

我才不会告诉你们我快读不开long long见祖宗了

还有就是要通过\(ans=(ans+M)\bmod M\)来保证\(ans\)是正的

上代码,因为不开long long那事调了好长时间。。

P4777 【模板】扩展中国剩余定理(EXCRT)

#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline LL read(){
LL x=0,y=1;
char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
int n;
LL a[100006],m[100006];
inline LL mul(LL n,LL k,LL mod){
LL ans=0;
while(k){
if(k&1) ans=(ans+n)%mod;
k>>=1;
n=(n+n)%mod;
}
return ans;
}
LL exgcd(LL a,LL b,LL &x,LL &y){
if(!b){x=1;y=0;return a;}
LL ret=exgcd(b,a%b,x,y);
LL z=x;x=y;y=z-(a/b)*y;
return ret;
}
inline LL excrt(){
LL x,y;
LL M=m[1],ans=a[1];
for(reg int i=2;i<=n;i++){
LL b=((a[i]-ans)%m[i]+m[i])%m[i];
LL gcd=exgcd(M,m[i],x,y);
x=mul(x,b/gcd,m[i]);
ans+=M*x;
M*=m[i]/gcd;
ans=(ans+M)%M;
}
return ans;
}
int main(){
n=read();
for(reg int i=1;i<=n;i++) m[i]=read(),a[i]=read();
std::printf("%lld",excrt());
return 0;
}

 

话说去年暑假在洛谷网校就学过一遍crt和excrt了

但当时就没怎么理解清楚,更写不出代码

这次是因为扩展卢卡斯要用到crt,才来写了一遍这两个题。。。

P1495 CRT,P4777 EXCRT的更多相关文章

  1. 「算法笔记」CRT 与 exCRT

    一.扩展欧几里得 求解方程 \(ax+by=\gcd(a,b)\). int exgcd(int a,int b,int &x,int &y){ if(!b) return x=1,y ...

  2. CRT和EXCRT学习笔记

    蒟蒻maomao终于学会\(CRT\)啦!发一篇博客纪念一下(还有防止忘掉) \(CRT\)要解决的是这样一个问题: \[x≡a_1​(mod m_1​)\] \[x≡a_2​(mod m_2​)\] ...

  3. CRT和EXCRT简单学习笔记

    中国剩余定理CRT 中国剩余定理是要求我们解决这样的一类问题: \[\begin{cases}x\equiv a_1\pmod {b_1} \\x\equiv a_2 \pmod{b_2}\\...\ ...

  4. 浅析中国剩余定理(从CRT到EXCRT))

    前置知识 1. a%b=d,c%b=e, 则(a+c)%b=(d+e)%b(正确性在此不加证明) 2. a%b=1,则(d\(\times\)a)%b=d%b(正确性在此不加证明) 下面先看一道题(改 ...

  5. Algorithm: CRT、EX-CRT & Lucas、Ex-Lucas

    中国剩余定理 中国剩余定理,Chinese Remainder Theorem,又称孙子定理,给出了一元线性同余方程组的有解判定条件,并用构造法给出了通解的具体形式. \[ \begin{aligne ...

  6. 关于一次同余方程的一类解法(exgcd,CRT,exCRT)

    1.解同余方程: 同余方程可以转化为不定方程,其实就是,这样的问题一般用拓展欧几里德算法求解. LL exgcd(LL a,LL b,LL &x,LL &y){ if(!b){ x=; ...

  7. 中国剩余定理(CRT)及其扩展(EXCRT)详解

    问题背景   孙子定理是中国古代求解一次同余式方程组的方法.是数论中一个重要定理.又称中国余数定理.一元线性同余方程组问题最早可见于中国南北朝时期(公元5世纪)的数学著作<孙子算经>卷下第 ...

  8. 中国剩余定理(CRT)

    只看懂了CRT,EXCRT待补.... 心得:记不得这是第几次翻CRT了,每次都有迷迷糊糊的.. 中国剩余定理用来求解类似这样的方程组: 求解的过程中用到了同余方程. x=a1( mod x1) x= ...

  9. $NOIp2018$劝退记

    鸽子博主好久没更博了,这一更可能以后都更不了了啊 \(Day~~1\) 考试爆零,已经无所畏惧了. 当作攒rp吧...qwq 晚上写了写数学总结,蒯了一堆人的博客资料,然后就学会了\(CRT\),\( ...

随机推荐

  1. MTK Android Framework用SystemProperties通过JNI调用访问系统属性

    1.导包 import android.os.SystemProperties; 2. Android SystemProperties设置/读取 #设置 Systemproperties.set(n ...

  2. Python——详解__str__, __repr__和__format__

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是Python专题的第10篇文章,我们来聊聊Python当中的类. 打印实例 我们先从类和对象当中最简单的打印输出开始讲起,打印一个实例 ...

  3. 22.3 Extends 构造方法的执行顺序

    /** 1.有子父类继承关系的类中,创建父类对象未调用,执行父类无参构造* 2.有子父类继承关系的类中,创建子类对象未调用,执行顺序:默认先调用 父类无参构造---子类无参构造* 在子类的构造方法的第 ...

  4. 10.6 IoStudentManager

    package day11_io_student.student_demo; public class Student { private String id; private String name ...

  5. django创建app

    前几天,

  6. 正整数的二进制表示中1的个数计算(使用移位或者n&(n-1))

    第一种:使用n&(n-1)表示来计算有多少个1 int n=127; int count=0; while (n!=0){ count++; n=n&(n-1); } 第二种:使用移位 ...

  7. CVE-2019-17671:wrodpress 未授权访问漏洞-复现

    0x00 WordPress简介 WordPress是一款个人博客系统,并逐步演化成一款内容管理系统软件,它是使用PHP语言和MySQL数据库开发的,用户可以在支持 PHP 和 MySQL数据库的服务 ...

  8. D3.js 力导向图的拖拽(drag)与缩放(zoom)

    不知道大家会不会跟我一样遇到这样的问题,在之前做的力导向图的基础上加上缩放功能的时候,拖动节点时整体会平移不再是之前酷炫的效果(失去了拉扯的感觉!).天啊,简直不能接受如此丑X的效果.经过不懈的努力终 ...

  9. Feature list, Standard and Test plan for BETA Release 12/22/2015

    ===================BETA RELEASE FEATRURE LIST==================== 1. Log in and account manager for ...

  10. C语言二维数组超细讲解

    用一维数组处理二维表格,实际是可行的,但是会很复杂,特别是遇到二维表格的输入.处理和输出. 在你绞尽脑汁的时候,二维数组(一维数组的大哥)像电视剧里救美的英雄一样显现在你的面前,初识数组的朋友们还等什 ...