中国剩余定理及其拓展 CRT&EXGCD
中国剩余定理,又叫孙子定理。
作为一个梗广为流传。其实它的学名叫中国单身狗定理。
- 中国剩余定理
中国剩余定理是来干什么用的呢?
其实就是用来解同余方程组的。那么什么又是同余方程组呢。
顾名思义就是n个同余方程。
形如
如果只有一个方程的话那么是很容易用exgcd来解决。
但如果变成n个就需要用到CRT了。
下面我们言归正传。
首先我们要知道只有满足m1,m2,mn两两互质才能运用CRT。
首先,我们令M=Πni=1。
令Mi=M/mi,这样我们就可以满足Mi%mk=0(k!=i)。
然后我们在构造n个数,对于每一个数ti满足Mi×ti≡1(mod mi),即ti为Mi在mod mi意义下的乘法逆元(好像是这么叫吧),用exgcd可求。
最后再把所有Mi×ti×ai求和再取模就可以了。
为什么这样做是正确的呢?因为Mi×ti≡1(mod mi),则Mi×ti×ai≡ai(mod mi)。
显然,把所有的数加到一起必定满足方程。(切记不要忘了取模)。
一道模板题LOJ10212
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <queue>
#include <vector>
using namespace std;
int m[], a[];
long long qpow(long long a, long long b) {
long long ans = ;
while (b) {
if (b & )
ans = ans * a;
b = b >> ;
a = a * a;
}
return ans;
}
long long exgcd(long long a, long long b, long long &x, long long &y) {
long long t;
if (!b) {
x = ;
y = ;
return a;
}
long long gcd = exgcd(b, a % b, x, y);
t = x;
x = y;
y = t - a / b * y;
return gcd;
}
int main() {
int n;
long long M = ;
scanf("%d", &n);
for (int i = ; i <= n; i++) {
scanf("%d%d", &m[i], &a[i]);
M *= 1ll * m[i];
}
long long ans = ;
for (int i = ; i <= n; i++) {
long long x, y;
long long Mi = M / m[i];
exgcd(Mi, m[i], x, y);
ans += x * Mi * a[i];
}
printf("%lld", (ans % M + M) % M);
}
CRT
- 拓展中国剩余定理
在上面CRT讲解中我们提到所有的m必须两两互质,因为如果不两两互质就可能无解或求错。
但是如果毒瘤出题人就让他们不互质呢。
那我们就需要把出题人吊起来打一顿用到拓展CRT(实际上他和CRT没有任何关系)
首先只考虑有两个方程怎么做
x=a1+k1×m1,x=a2+k2×m2 。可以得到k2×m2-k1×m1=a1-a2。
嗯,看起来有点像一个二元一次方程了。
我们令 gcd=gcd(m1,m2),c=a1-a2;
如果gcd%c!=0则无解
否则用exgcd来求出k1×m1+k2×m2=gcd的解k1,再乘上c/gcd就好了。
为了避免爆long long 最好取一下模。
这样我们就可以反推出x。
但注意我们解的方程k1项系数为-1,相当与k1=-k1(所以k1=0?)
这样就求出了x。(可以把这个x0转化成最小的非负数解)
这个x符合第一个方程,也符合第二个方程。设这个x为x0
所以,可以得到通解是:x=x0+k×lcm(m1,m2)
满足这个条件的x就满足第一第二两个方程。满足第一第二两个方程的所有的解也都是这个方程的解。
所以第一第二个方程和这个方程是等价的。
将这个方程转化一下,可以得到新的同余方程:x=x0(modlcm(m1,m2))
这样,我们成功的把两个方程转化成了一个方程,以此类推。
最后留下的这个方程,它的x_0的最小非负数解,就是我们要的最终答案(最后这一段用文本编辑器打的,略显不符,请自动忽略)。
例题 poj2891
模板题,直接放代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<vector>
#include<cstdlib>
using namespace std;
#define int long long
int m[],c[];
int exgcd(int a,int b,int &x,int &y){
int t;
if(!b){
x=;y=;
return a;
}
int gcd=exgcd(b,a%b,x,y);
t=x;
x=y;
y=t-a/b*y;
return gcd;
}
signed main(){
int n;
while(~scanf("%lld",&n)){
bool ju=;
for(int i=;i<=n;i++) scanf("%lld%lld",&m[i],&c[i]);
int m1=m[],c1=c[];
for(int i=;i<=n;i++){
int x,y;
int m2=m[i],c2=c[i];
int dembele=exgcd(m1,m2,x,y);
if((c2-c1)%dembele){
ju=;
printf("-1\n");
break;
}
x=(c1-c2)/dembele*x%m2;
c1-=m1*x;
m1=m1/dembele*m2;
c1%=m1;
}
if(ju)
printf("%lld\n",(c1%m1+m1)%m1);
}
}
EXCRT
中国剩余定理及其拓展 CRT&EXGCD的更多相关文章
- 扩展中国剩余定理(扩展CRT)详解
今天在$xsy$上翻题翻到了一道扩展CRT的题,就顺便重温了下(扩展CRT模板也在里面) 中国剩余定理是用于求一个最小的$x$,满足$x\equiv c_i \pmod{m_i}$. 正常的$CRT$ ...
- 模线性方程&&中国剩余定理及拓展
一.求解模线性方程 由ax=b(mod n) 可知ax = ny + b 就相当于ax + ny = b 由扩展欧几里得算法可知有解条件为gcd(a, n)整除d 可以直接套用扩展欧几里得算法 最终由 ...
- 浅析中国剩余定理(从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(正确性在此不加证明) 下面先看一道题(改 ...
- E - Two Arithmetic Progressions(CodeForces - 710D)(拓展中国剩余定理)
You are given two arithmetic progressions: a1k + b1 and a2l + b2. Find the number of integers x such ...
- 中国剩余定理(CRT)及其扩展(EXCRT)详解
问题背景 孙子定理是中国古代求解一次同余式方程组的方法.是数论中一个重要定理.又称中国余数定理.一元线性同余方程组问题最早可见于中国南北朝时期(公元5世纪)的数学著作<孙子算经>卷下第 ...
- 欧几里得(辗转相除gcd)、扩欧(exgcd)、中国剩余定理(crt)、扩展中国剩余定理(excrt)简要介绍
1.欧几里得算法(辗转相除法) 直接上gcd和lcm代码. int gcd(int x,int y){ ?x:gcd(y,x%y); } int lcm(int x,int y){ return x* ...
- 中国剩余定理(CRT)及其拓展(ExCRT)
中国剩余定理 CRT 推导 给定\(n\)个同余方程 \[ \left\{ \begin{aligned} x &\equiv a_1 \pmod{m_1} \\ x &\equiv ...
- 拓展中国剩余定理(exCRT)摘要
清除一个误区 虽然中国剩余定理和拓展中国剩余定理只差两个字,但他俩的解法相差十万八千里,所以会不会CRT无所谓 用途 求类似$$\begin{cases}x \equiv b_{1}\pmod{a_{ ...
- POJ.1006 Biorhythms (拓展欧几里得+中国剩余定理)
POJ.1006 Biorhythms (拓展欧几里得+中国剩余定理) 题意分析 不妨设日期为x,根据题意可以列出日期上的方程: 化简可得: 根据中国剩余定理求解即可. 代码总览 #include & ...
随机推荐
- mysql45讲
一共48讲,学习时间2天. 开篇词 (1讲) 开篇词 | 这一次,让我们一起来搞懂MySQL 理论指导实践:先系统性的学习原理,再实践验证. 你可以从点到线再到面,形成自己到mysql知识网络. 在使 ...
- MVC4学习要点记三
一.数据迁移用来解决code first情况下当增加.删除.改变实体类,或改变DbContext类后,相应地更新数据库结构而不会对现有数据产生影响. 1.启用迁移指令:enable-migration ...
- shiro学习(四、shiro集成spring+springmvc)
依赖:spring-context,spring-MVC,shiro-core,shiro-spring,shiro-web 实话实说:web.xml,spring,springmvc配置文件好难 大 ...
- shell、bash、terminal和kernel之间的关系
shell.bash.terminal和kernel之间的关系 本文是<鸟哥的Linux私房菜>的学习笔记 什么是kernel kernel的中文是"操作系统核心",主 ...
- java enum类自定义属性
enum类自定义属性 这就是enum比static静态变量好用的地方了,可以赋予每一个枚举值若干个属性,例如 实例1: public enum GasStationChannel { ZH(" ...
- Python学习计划-首篇
概述 Python语言这几年的热度一直持续上升,有多火也不用我再介绍了,光从近几年编程语言排行榜的上升热度就能看出,因此我也计划学习一下python语言,并且将每次的学习收获记录下来.一方面是记录下来 ...
- Phoenix批量提交优化,官网的demo
1 Phoenix的批量insert官网代码,最佳实践 try (Connection conn = DriverManager.getConnection(url)) { conn.setAutoC ...
- 第十四章·Kibana深入-Timelion画图实现系统监控
什么是Timelion? Timelion使你可以轻松获得以下问题的答案: 1)随着时间的推移,每个唯一的用户会查看多少个页面?2)这个星期五和上周五之间的交通量有什么不同?3)今天有多少日本人口来到 ...
- Git学习笔记(1)
一.Git特点 1.直接快照,而非比较差异 Git只关心文件数据的整体是否发生变化,而不关心具体文件及其内容发生了那些变化.也就是说Git在保存文件时,每次只会在上一次版本基础上保存那些变化的文件,为 ...
- QTP(7)
一.输出值(Output Value) 1.应用场景: 1) 关心被测系统的数据 2) 将被测系统生成的数据作为后面步骤的输入 2.输出值就是输出被测系统中实际运行时的数据的一种技术 a.运行中对象的 ...