UOJ#42. 【清华集训2014】Sum 类欧几里德算法

原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ42.html

题解

首先我们把式子改写一下:

$$(-1)^{\lfloor a\rfloor} \\=1-2(\lfloor a\rfloor \bmod 2)\\=1-2(\lfloor a\rfloor -2\lfloor \frac a2 \rfloor)$$

于是问题就变成了求解:

$$f(a,b,c,n) = \sum_{i=1}^n \left\lfloor \frac {a\sqrt{r} +b}{c}i\right\rfloor$$

按照类欧几里得算法的思路,我们把他变成一个 二维坐标系中  数梯形内整点 的问题,通过不断翻转坐标系搞一搞。

首先求出 $\left\lfloor \frac {a\sqrt{r} +b}{c}\right\rfloor$ 的值,即梯形短的一个底边的长度下取整。

然后把梯形转化成一个三角形。

然后把坐标系按照直线 $y=x$ 翻转,那么斜率取倒数:

$$\frac c {a\sqrt{r} + b} = \frac{c(a\sqrt r -b)}{a^2r-b^2} = \frac {ac\sqrt r -bc}{a^2r-b^2}$$

然后像类欧一样递归下去就好了。

代码

#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
#define int long long
using namespace std;
typedef long long LL;
LL read(){
	LL x=0,f=0;
	char ch=getchar();
	while (!isdigit(ch))
		f|=ch=='-',ch=getchar();
	while (isdigit(ch))
		x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
	return f?-x:x;
}
int T,n,r;
double rt;
int gcd(int a,int b){
	return b?gcd(b,a%b):a;
}
int f(int a,int b,int c,int n){
	if (!n)
		return 0;
	int t=gcd(a,gcd(b,c));
	a/=t,b/=t,c/=t;
	double k=1.0*(rt*a+b)/c;
	int kk=(int)k;
	k-=kk;
	int m=(int)(k*n);
	b-=c*kk;
	return n*m+kk*(n+1)*n/2-f(a*c,-b*c,a*a*r-b*b,m);
}
signed main(){
	T=read();
	while (T--){
		n=read(),r=read();
		rt=sqrt(r);
		int t=(int)rt;
		if (t*t==r){
			if (r&1)
				puts(n&1?"-1":"0");
			else
				printf("%lld\n",n);
		}
		else
			printf("%lld\n",n-2*f(1,0,1,n)+4*f(1,0,2,n));
	}
	return 0;
}

  

posted @ 2019-02-20 20:59  zzd233  阅读(301)  评论(1编辑  收藏  举报