题目会给你两组数字,分别是n和a,求最小的x。
如果所有的n1,n2,.....nk两两互质,那么就可以用中国剩余定理,否则就是扩展中国剩余定理。
中国剩余定理:
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define IOS ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
const int N=1e5+5;
int n,m;
int a[N],b[N];
int exgcd(int a,int b,int &x,int &y){
if(b==0){
x=1,y=0;
return a;
}
int gcd=exgcd(b,a%b,y,x);
y-=(a/b)*x;
return gcd;
}
int CRT(){
int p=1,ans=0;
for(int i=1;i<=m;i++) p*=a[i];
for(int i=1;i<=m;i++){
int c=p/a[i],x=0,y=0;
exgcd(c,a[i],x,y);
ans=(ans+b[i]*c*x%p)%p;
}
return (ans%p+p)%p;
}
void solve(){
cin>>n;
for(int i=1;i<=n;i++){
cin>>a[i]; //模数
}
for(int i=1;i<=n;i++){
cin>>b[i]; //余数
}
cout<<CRT()<<endl;
}
signed main(){
IOS
int t=1;
cin>>t;
while(t--){
solve();
}
return 0;
}
拓展中国剩余定理:
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define IOS ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
const int N=1e5+5;
int n,m;
int a[N],b[N];
int exgcd(int a,int b,int &x,int &y){
if(b==0){
x=1,y=0;
return a;
}
int gcd=exgcd(b,a%b,y,x);
y-=(a/b)*x;
return gcd;
}
int CRT(){
int p=a[1],r=b[1],x=0,y=0;
for(int i=2;i<=m;i++){
int c=b[i]-r,gcd=exgcd(p,a[i],x,y);
if(c%gcd){
f=1;
return 0;
}
int tmp=c/gcd*x,t=a[i]/gcd;
tmp=(tmp%t+t)%t;
r+=p*tmp;
p=a[i]/gcd*p;
}
if(r==0) return p;
return r;
}
void solve(){
cin>>n;
for(int i=1;i<=n;i++){
cin>>a[i]; //模数
}
for(int i=1;i<=n;i++){
cin>>b[i]; //余数
}
cout<<CRT()<<endl;
}
signed main(){
IOS
int t=1;
cin>>t;
while(t--){
solve();
}
return 0;
}