zoukankan      html  css  js  c++  java
  • POJ 2115 C Looooops (扩展欧几里德 + 线性同余方程)

      分析:这个题主要考察的是对线性同余方程的理解,根据题目中给出的a,b,c,d,不难的出这样的式子,(a+k*c) % (1<<d) = b; 题目要求我们在有解的情况下求出最小的解,我们转化一下形式。

      上式可以用同余方程表示为  a + k*c = (b) % (1<<d)   <-->  k*c = (b-a) % (1<<d)(中间应该是全等号,打不出来…)。这就是我们想要的同余方程,根据我的个人习惯,我把它转化为线性方程的形式。

      -->   c*x + (1<<d)*y = (b-a); 此时就对应了线性方程中的a*x + b*y = c的形式,当且仅当 c % gcd(a,b) == 0时x有解,得到x的一个解以后,我们另x化为(x%mod + mod) % mod的形式,mod = b/gcd(a,b);  这个时候的x范围处于(1,mod-1)之间,正是我们要的最小解。

      注意:注意LL的形式,在(1<<d)的时候,不要忘记改成(1LL << d),否则WA了,我被这里坑了……

      代码如下:

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    using namespace std;
    #define LL long long
    /**
    
    a + c*k = b % d
    -> c*k = (b-a) % d
    -> c*x + d*y = k*(b-a);
    -> if gcd(c,d) | (b-a);
    -> get x = k;
    -> else forever.
    */
    LL x,y;
    LL exgcd(LL a,LL b){
        if(!b){
            x = 1; y = 5;
            return a;
        }
        LL gcd = exgcd(b,a%b);
        LL tmp = x;
        x = y;
        y = tmp - a/b * y;
        return gcd;
    }
    int main(){
        LL a,b,c,d;
        while(~scanf("%I64d %I64d %I64d %I64d",&a,&b,&c,&d)){
            if(a+b+c+d == 0) break;
            d = (1LL<<d);
            LL A,B,gcd,C,ans,k,mod;
            A = c; B = d; C = (b-a);
            gcd = exgcd(A,B);
            if(C % gcd != 0){
                puts("FOREVER");
            }
            else {
                k = C / gcd;
                x = k*x;
                mod = B / gcd;
                ans = (x%mod + mod) % mod;
                printf("%I64d
    ",ans);
            }
        }
        return 0;
    
    }
  • 相关阅读:
    汪曾祺《生活是很好玩的》有感
    中小学Python编程语言教学
    1. python 基础课程目录
    白盒测试之gmock入门篇
    Apache与Tomcat 区别联系
    白盒测试之gtest第一个demo
    白盒测试之初识gtest工具
    纯手工打造dropdownlist控件
    android 线程的使用总结
    染色板QPlette
  • 原文地址:https://www.cnblogs.com/jifahu/p/5574024.html
Copyright © 2011-2022 走看看