[BZOJ 2299][HAOI 2011]向量 题解(裴蜀定理)

[BZOJ 2299][HAOI 2011]向量

Description

给你一对数a,b,你能够任意使用(a,b), (a,-b), (-a,b), (-a,-b), (b,a), (b,-a), (-b,a), (-b,-a)这些向量,问你能不能拼出另外一个向量(x,y)。
说明:这里的拼就是使得你选出的向量之和为(x,y)ios

Input
第一行数组组数t,(t<=50000)
接下来t行每行四个整数a,b,x,y (-2109<=a,b,x,y<=2109)git

Output
t行每行为Y或者为N,分别表示能够拼出来,不能拼出来数组

Solution

1.考虑把八种状况合在一块儿,发现反向操做能够合并,那么操做也就是有四种(a,b),(-a,b),(b,a),(-b,a);spa

2.设四种操做进行的次数分别为x1,x2,x3,x4,那么:code

对横坐标的操做为x1a-x2a+x3b-x4b,即(x1-x2)a+(x3-x4) b=x;ip

对纵坐标的操做为x1b+x2b+x3a+x4a,即(x1+x2)b+(x3+x4)a=y;get

因而咱们验证两方程是否有整数解便可。string

3.裴蜀定理告诉咱们:当gcd(a,b)|ans时,方程有整数解。it

考虑四个系数的奇偶性,由于四个系数是四个数的组合,因此gcd应该是原来的二倍。io

好比当x1-x2是奇数时,假设可行解中系数x1+x2为偶数,那就不成立了,因此咱们分状况讨论以后发现,知足要求时gcd应该为原来的二倍。

Code

#include<iostream>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
typedef long long ll;
using namespace std;
 
ll g;
 
inline ll rd(){
    ll x=0; 
    bool f=0;
    char c=getchar();
    while(!isdigit(c)){
        if(c=='-')f=1;
        c=getchar();
    }
    while(isdigit(c)){
        x=(x<<1)+(x<<3)+(c^48);
        c=getchar();
    }
    return f?-x:x;
} 
 
ll gcd(ll x,ll y){return y?gcd(y,x%y):x;} 
 
bool valid(ll x,ll y){return (!(x%g))&&(!(y%g));}
 
int main(){
    ll t=rd();
    while(t--){
        ll a=rd(),b=rd();
        ll x=rd(),y=rd();
        g=gcd(a,b)<<1;
        printf((valid(x,y)||valid(x+a,y+b)||valid(x+b,y+a)||valid(x+a+b,y+a+b))?"Y\n":"N\n");
    }
    return 0;
}
相关文章
相关标签/搜索