exgcd求解同余方程的最小正整数解 poj1061 poj2115

这两题都是求解同余方程,并要求出最小正整数解的

对于给定的Ax=B(mod C) 要求x的最小正整数解

首先这个式子可转化为 Ax+Cy=B,那么先用exgcd求出Ax+Cy=gcd(A,C)的解x

然后这个式子的一个特解就是 (B/gcd(A,C))* x

要注意如果gcd(A,C)无法整除B,那么这个式子无解

然后是求出最小整数解

Ax+Cy=B 方程的通解是 x+k*C/gcd(A,C),

另s=C/gcd(A,C)

所以最小整数解是(x%s+s)%s

青蛙题

/*
x+km=y+kn(mod L)

x+km+t*L=y+kn
k(m-n)+(x-y)=-t*L
k(n-m)-(x-y)=t*L

要求出k
即(n-m)k = x-y(mod L)  
那么(x-y)%gcd(n-m,L)==0
 
 如果(x-y)%gcd(n-m,L)!=0,那么就是无解
  
用exgcd(n-m,l,x,y)出k0即可 
答案是(x-y)/gcd(n-m,L)*k0+l/gcd(n-m,l) 
*/
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
#define ll long long 
ll x,y,m,n,L;
ll exgcd(ll a,ll b,ll &x,ll &y){
    if(b==0){x=1;y=0;return a;}
    ll d=exgcd(b,a%b,y,x);
    y-=a/b*x;
    return d;
}
int main(){
    while(cin>>x>>y>>m>>n>>L){
        ll a=n-m,b=L,c=x-y,k,u;
        ll d=exgcd(a,b,k,u);
        if(c%d!=0){
            puts("Impossible");
            continue;
        }
        k=k*(c/d);//该方程特解
        ll m=b/d;
        printf("%lld\n",(k%m+m)%m); 
    }
}

looop题

/*
Cx=B-A(mod L)
转化为求同余方程
Cx+Ly=B-A
d=gcd(C,L)
d|B-A,则有解
exgcd(C,L,x,y)解得x 
先求出一个特解x=(B-A)/d*x
然后再求最小整数解即可 
*/
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
#define ll long long
ll exgcd(ll a,ll b,ll &x,ll &y){
    if(!b){x=1,y=0;return a;}
    ll d=exgcd(b,a%b,y,x);
    y-=a/b*x;
    return d;
}
int main(){
    ll A,B,C,k,L;
    while(cin>>A>>B>>C>>k && (A||B||C||k) ){
        L=1;
        for(int i=1;i<=k;i++)
            L*=2;
        ll d,x,y;
        d=exgcd(C,L,x,y); 
        if((B-A)%d!=0){
            puts("FOREVER");
            continue;
        }
        
        x=(B-A)/d*x;
        ll s=L/d;
        cout<<(x%s+s)%s<<endl;
    }    
} 

猜你喜欢

转载自www.cnblogs.com/zsben991126/p/10443172.html