题目:
http://codeforces.com/gym/101498/problem/G
题意:
给n个数,求有多少对[l,r]使得∑data[i] (i∈[l,r]) 能被任一data[j]整除 (j∈[l,r])
分析:
设一堆数的和为sum,这堆数的最小公倍数为lcm;
若这堆数的和,能被任一数整除,则sum%lcm==0;
(理解:最小的能被任一数整除的数就是lcm,则这堆数的和一定要为lcm的倍数才行)
一堆数的lcm怎么求呢?
假设已知前n-1个数的lcm为t;
那么这n个数的lcm就是:t*data[n]/gcd(t,data[n]);
So,O(n^2*logn)循环就行。
代码:
#include <bits/stdc++.h>
using namespace std;
typedef unsigned long long ll;
const int tmax=2005;
int n,ans;
ll data[tmax],sum[tmax],lcm;
ll gcd(ll a,ll b)
{
if(b==0) return a;
return gcd(b,a%b);
}
int main()
{
int T,i,j;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
ans=n;
for(i=1;i<=n;i++)
{
scanf("%I64d",&data[i]);
sum[i]=sum[i-1]+data[i];
}
for(i=1;i<=n;i++) //左端点
{
lcm=data[i];
for(j=i+1;j<=n;j++) //右端点
{
lcm=lcm*data[j]/gcd(lcm,data[j]);
if((sum[j]-sum[i-1])%lcm==0) ans++;
if(lcm>(sum[n]-sum[i-1])) break;
}
}
printf("%d\n",ans);
}
return 0;
}