给你一对数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,分别表示能够拼出来,不能拼出来数组
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应该为原来的二倍。
#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; }