URAL 2032 - Conspiracy Theory and Rebranding【本源勾股数组】

【题意】

给出三角形的三个边长,均是10^7以内的整数,问三角形的三个角的坐标是否能均是整数,输出其中任意一个解。

 

 【题解】

一开始想的是枚举一条边的横坐标,然后通过勾股定理以及算角度求出其他点的坐标,再判断是否符合条件。

亲测TLE

 

直到知道了本源勾股数组的构造方法。。。

每个本源勾股数组(a,b,c)满足a*a+b*b=c*c,其中a为奇数,b为偶数。。

枚举s,t(1<=t<s,且它们是没有公因数的奇数) 

a=st  b=(s*s-t*t)/2  c=(s*s+t*t)/2

因为最大数c=(s*s+t*t)/2  所以最多枚举到sqrt(2*c)即可。

 

假设三角形的三个点分别为p,q和r

我们先固定一个点为p(0,0),另外一个点q与它的距离是x,还有一个点r与它的距离是y。那么q的距离与r的距离一定是z

我们枚举勾股数组,如果勾股数组(a1,b1,c1)的c1,也就是最大的那个数,等于x,那么x的坐标为(a1,b1)【当然也可以是(a1,-b1),(-a1,b1),(-a1,-b1),均需要枚举,下同】

然后枚举c等于y的勾股数组,(a2,b2,c2),那么r点坐标为(a2,b2) 【可以事先把这些坐标预处理出来,放入vector中】

接下来判断两坐标是否相距为z即可。

 

注意通过这种方法求出来的勾股数组的a是奇数,也就是说它们的倍数 (i*a,i*b,i*c),i是一个正整数,并不会被求出来,我们要求的是i*c==x,那么只要满足x mod c=0我们就可以把勾股数组乘以x/c,加入备选选项中。

注意(0,x) (0,-x) (x,0) (-x,0)以及(0,y) (0,-y) (y,0) (-y,0) 不会在枚举本源勾股数组中出现,所以需要自己手动判断。

 

#include<bits/stdc++.h>

#define eps 1e-9

#define FOR(i,j,k) for(int i=j;i<=k;i++)

#define MAXN 1005

#define MAXM 40005

#define INF 0x3fffffff

#define PB push_back

#define MP make_pair

#define X first

#define Y second

#define lc (k<<1)

#define rc ((k<<1)1)

using namespace std;

typedef long long LL;

LL i,j,k,n,m,x,y,T,ans,big,cas,num,len;

bool flag;

LL z;

LL mx,sum,a,b,c;

vector <pair<LL,LL> > xx,yy;



LL gcd(LL x, LL y)

{

    return y ? gcd(y, x % y) : x;

}



int main()

{

    scanf("%I64d%I64d%I64d",&x,&y,&z);

    if (x>y) swap(x,y);

    if (y>z) swap(y,z);

    if (x>y) swap(x,y);

    mx=(LL)(sqrt(2*z)+eps);

    

    for (i=1;i<=mx;i+=2)//枚举本源勾股数组

    {

        for (j=i+2;j<=mx;j+=2)

        {

            if (gcd(i,j)>1) continue;

            a=i*j;

            b=(j*j-i*i)/2;

            c=(j*j+i*i)/2;

            if (x%c==0)

            {

                xx.PB(MP(a*x/c,b*x/c));

                xx.PB(MP(a*x/c,-b*x/c));

                xx.PB(MP(-a*x/c,b*x/c));

                xx.PB(MP(-a*x/c,-b*x/c));

                xx.PB(MP(b*x/c,a*x/c));

                xx.PB(MP(b*x/c,-a*x/c));

                xx.PB(MP(-b*x/c,a*x/c));

                xx.PB(MP(-b*x/c,-a*x/c));

            }

            if (y%c==0) 

            {

                yy.PB(MP(a*y/c,b*y/c));

                yy.PB(MP(a*y/c,-b*y/c));

                yy.PB(MP(-a*y/c,b*y/c));

                yy.PB(MP(-a*y/c,-b*y/c));

                yy.PB(MP(b*y/c,a*y/c));

                yy.PB(MP(b*y/c,-a*y/c));

                yy.PB(MP(-b*y/c,a*y/c));

                yy.PB(MP(-b*y/c,-a*y/c));

            }

        }

    }

    xx.PB(MP(0,x));xx.PB(MP(x,0));xx.PB(MP(0,-x));xx.PB(MP(-x,0));

    yy.PB(MP(0,y));yy.PB(MP(y,0));yy.PB(MP(0,-y));yy.PB(MP(-y,0));

    

    for (i=0;i<xx.size();i++)

    {

        for (j=0;j<yy.size();j++)

        {

            if ((xx[i].X-yy[j].X)*(xx[i].X-yy[j].X)+(xx[i].Y-yy[j].Y)*(xx[i].Y-yy[j].Y)==z*z)

            {

                printf("0 0\n%I64d %I64d\n%I64d %I64d\n",xx[i].X,xx[i].Y,yy[j].X,yy[j].Y);

                return 0;

            }

        }

    }

    printf("-1\n");

    return 0;

}

 

你可能感兴趣的:(RAC)