UVA - 11768 Lattice Point or Not (扩展欧几里得)

求一条线段上有多少个整点。

是道扩欧基础题,列出两点式方程,然后分四种情况讨论即可。但细节处理较多很容易写挫(某zzWA了十几发才过掉的)。

由于数据精度较小,浮点数比较没有用eps,直接==比较了。

 1 #include<bits/stdc++.h>
 2 
 3 using namespace std;
 4 typedef long long ll;
 5 
 6 void exgcd(ll a,ll b,ll& x,ll& y,ll& g) {
 7     if(!b)x=1,y=0,g=a;
 8     else exgcd(b,a%b,y,x,g),y-=x*(a/b);
 9 }
10 
11 ll solve(double x1,double y1,double x2,double y2) {
12     if(x1>x2||(x1==x2&&y1>y2))swap(x1,x2),swap(y1,y2);
13     if(x1==x2&&y1==y2) {
14         if(x1==round(x1)&&x2==round(x2))return 1;
15         else return 0;
16     } else if(x1==x2) {
17         if(x1==round(x1))return floor(y2)-ceil(y1)+1;
18         else return 0;
19     } else if(y1==y2) {
20         if(y1==round(y1))return floor(x2)-ceil(x1)+1;
21         else return 0;
22     } else {
23         ll a=round((y2-y1)*100),b=round((x1-x2)*100),c=round((y2*x1-y1*x2)*100);
24         ll x,y,g;
25         exgcd(a,b,x,y,g);
26         if(c%g==0) {
27             x*=c/g;
28             ll bb=abs(b/g);
29             ll xl=ceil(x1),xr=floor(x2);
30             ll xll=x+(xl-x)/bb*bb;
31             ll xrr=x+(xr-x)/bb*bb;
32             if(xll<xl)xll+=bb;
33             if(xrr>xr)xrr-=bb;
34             return max(0ll,(xrr-xll)/bb+1);
35         } else return 0;
36     }
37 }
38 
39 int main() {
40     ll T;
41     scanf("%lld",&T);
42     while(T--) {
43         double x1,y1,x2,y2;
44         scanf("%lf%lf%lf%lf",&x1,&y1,&x2,&y2);
45         printf("%lld\n",solve(x1,y1,x2,y2));
46     }
47     return 0;
48 }

 

posted @ 2019-02-06 19:54  jrltx  阅读(250)  评论(0编辑  收藏  举报