bzoj1041[HAOI2008]圆上的整点

bzoj1041[HAOI2008]圆上的整点

题意:

求一个给定半径的圆圆周上有多少个点的坐标是整数。r≤2*109

题解:

数学神题,本弱只能转载一下黄学长的题解

首先x²+y²=r²,变形得y²=(r+x)*(r-x)。令d=gcd(r+x,r-x),则A=(r-x)/d,B=(r+x)/d,A,B互质,用a,b代入,有y²=d²*A*B,由于d²,y²为完全平方数,故A*B也为完全平方数。又因为A≠B,所以A和B都是完全平方数。设a²=A=(r-x)/d,b²=B=(r+x)/d,则a²+b²=2r/d,因此d为r的约数。

有了上面的推理,那么实现的方法为:枚举d∈[1,sqrt(2R)],然后根据上述推理可知:必先判d是否为2R的一约数。此时d为2R的约数有两种情况:d=d或d=2R/d。

第一种情况:d=2R/d。枚举a∈[1,sqrt(2R/d/2)] ,算出对应的b=sqrt(2R/d-a^2),检查是否此时的A,B满足:A≠B且A,B互质,若是就将答案加1

第二种情况:d=d。枚举a∈[1,sqrt(d/2)],算出对应的b=sqrt(d-a^2),检查是否此时的A,B满足:A≠B且A,B互质 <根据上面的推理可知必需满足此条件>,若是就将答案加1

因为这样只算出了第一象限的情况<上面枚举时均是从1开始枚举>,根据圆的对称性,其他象限的整点数与第一象限中的整点数相同,最后,在象限轴上的4个整点未算,加上即可,那么最后答案为ans=4*第一象限整点数+4

代码:

 1 #include <cstdio>
 2 #include <algorithm>
 3 #include <cstring>
 4 #include <cmath>
 5 #define ll long long
 6 #define inc(i,j,k) for(ll i=j;i<=k;i++)
 7 using namespace std;
 8 
 9 ll gcd(ll a,ll b){return b==0?a:gcd(b,a%b);}
10 int main(){
11     ll n,ans=0; scanf("%lld",&n); n*=2;
12     inc(i,1,(ll)sqrt(n))if(n%i==0){
13         ll d=i;
14         inc(j,1,(ll)sqrt(n/d/2)){
15             ll b=n/d-j*j; if(sqrt(b)==(double)((ll)sqrt(b))&&gcd(j*j,b)==1&&j*j!=b)ans++;
16         }
17         if(d!=n/i){
18             d=n/i;
19             inc(j,1,(ll)sqrt(n/d/2)){
20                 ll b=n/d-j*j; if(sqrt(b)==(double)((ll)sqrt(b))&&gcd(j*j,b)==1&&j*j!=b)ans++;
21             }
22         }
23     }
24     printf("%lld",ans*4+4); return 0;
25 }

 

20160615

posted @ 2016-08-17 14:22  YuanZiming  阅读(230)  评论(0编辑  收藏  举报