BZOJ2749 HAOI2012外星人(数论)

  不妨把求φ抽象成把将每个位置上的一个小球左移一格并分裂的过程,那么即求所有球都被移到1号格子的步数。

  显然要达到1必须先到达2。可以发现每次分裂一定会分裂出2号位的球,因为2以外的质数一定是奇数。以及,每次移动至多将一个2号位的球移至1号位。

  于是我们只要数出每个位置能将几个球分裂至2号位就可以了。注意初始时若2号位没有球答案要+1。

  这个数数可以用线性筛搞定。n为质数则有f[n]=f[n-1],否则有f[n]=f[prime]+f[n/prime]。

  (挤进bzoj前1k了www

#include<iostream> 
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
int read()
{
    int x=0,f=1;char c=getchar();
    while (c<'0'||c>'9') {if (c=='-') f=-1;c=getchar();}
    while (c>='0'&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
    return x*f;
}
#define N 100010
int T,n,f[N],prime[N],cnt=0;
bool flag[N];
int main()
{
#ifndef ONLINE_JUDGE
    freopen("bzoj2749.in","r",stdin);
    freopen("bzoj2749.out","w",stdout);
    const char LL[]="%I64d\n";
#else
    const char LL[]="%lld\n";
#endif
    flag[1]=1;f[1]=1;
    for (int i=2;i<=N-10;i++)
    {
        if (!flag[i]) prime[++cnt]=i,f[i]=f[i-1];
        for (int j=1;j<=cnt&&prime[j]*i<=N-10;j++)
        {
            flag[prime[j]*i]=1;
            f[prime[j]*i]=f[prime[j]]+f[i];
            if (i%prime[j]==0) break;
        }
    }
    T=read();
    while (T--)
    {
        n=read();
        long long ans=1;
        for (int i=1;i<=n;i++)
        {
            int x=read(),y=read();
            ans+=1ll*y*f[x]-(x==2);
        }
        printf(LL,ans);
    }
    return 0;
}

 

posted @ 2018-09-08 15:10  Gloid  阅读(136)  评论(0编辑  收藏  举报