zoukankan      html  css  js  c++  java
  • loj#572. 「LibreOJ Round #11」Misaka Network 与求和

    胡策题,迫使懒得点技能树选手学算法。。。

    至少杜教和min_25会了

    不想写小结啊啊啊啊

    那么对于这个题先上个莫反套路

    然后用杜教乘个I把f*u给消掉,发现h就是f,这个可以用类似min_25的思想搞。。。

    #include<cstdio>
    #include<iostream>
    #include<cstring>
    #include<cstdlib>
    #include<algorithm>
    #include<cmath>
    #include<map>
    #include<ctime>
    using namespace std;
    typedef long long LL;
    typedef unsigned int uint;
    const int _=1e2;
    const int maxs=1e5+_;
    
    namespace MIN_25
    {
        int N,sqN;
        uint quick_pow(uint A,int p)
        {
            uint ret=1;
            while(p!=0)
            {
                if(p%2==1)ret=ret*A;
                A=A*A;p/=2;
            }
            return ret;
        }
        int pr,prime[maxs];uint kp[maxs]; bool v[maxs];
        void getprime(int K)
        {
            for(int i=2;i<=sqN;i++)
            {
                if(v[i]==false)prime[++pr]=i,kp[pr]=quick_pow(i,K);
                for(int j=1;j<=pr&&i*prime[j]<=sqN;j++)
                {
                    v[i*prime[j]]=true;
                    if(i%prime[j]==0)break;
                }
            }
        }
        
        int ulen,u[2*maxs],id1[maxs],id2[maxs];
        int ID(int x){return (x<=sqN)?id1[x]:id2[N/x];}
        uint g[2*maxs];//质数个数前缀和 
        void getg()
        {
            ulen=0;
            for(int i=1;i<=N;i=N/(N/i)+1)
            {
                u[++ulen]=N/i;
                if(u[ulen]<=sqN)id1[u[ulen]]=ulen;
                else id2[N/u[ulen]]=ulen;
                g[ulen]=u[ulen]-1;
            }
            
            for(int j=1;j<=pr;j++)
                for(int i=1;i<=ulen&&(LL)prime[j]*prime[j]<=u[i];i++)
                    g[i]-=g[ID(u[i]/prime[j])]-(j-1);
        }
        
        uint S(int n,int j)
        {
            if(n<=1||prime[j]>n)return 0;
            
            uint sum=0;
            for(int i=j;(LL)prime[i]*prime[i]<=n&&i<=pr;i++)
            {
                uint p=prime[i];
                for(int q=1;(LL)p*prime[i]<=n;q++)
                {
                    sum+=S(n/p,i+1);
                    sum+=kp[i]*(g[ID(n/p)]-(i-1));
                    p*=prime[i];
                }
            }
            if(j==1)sum+=g[ID(n)];
            return sum;
        }
    }
    
    namespace DJ
    {
        int N,sqN;
        uint mp1[maxs],mp2[maxs];
        uint MP(uint x){return (x<=sqN)?mp1[x]:mp2[N/x];}
        void upd(int x,uint d)
        {
            if(x<=sqN)mp1[x]=d;
            else mp2[N/x]=d;
        }
        uint S(int n)
        {
            if(MP(n)!=0)return MP(n);
            uint ret=MIN_25::S(n,1); int last;
            for(int i=2;i<=n;i=last+1)
            {
                last=n/(n/i);
                ret-=(last-i+1)*S(n/last);
            }
            upd(n,ret);
            return ret;
        }
    }
    
    int main()
    {
        int n,K;
        scanf("%d%d",&n,&K);
        
        DJ::N=MIN_25::N=n;
        DJ::sqN=MIN_25::sqN=int(sqrt(double(n+1)));
        MIN_25::getprime(K);
        MIN_25::getg();
        
        uint ans=0,ps=0,ns; int last;
        for(int i=2;i<=n;i=last+1)
        {
            last=n/(n/i);
            ns=DJ::S(last);
            ans+=(uint)(n/i)*(n/i)*(ns-ps);
            ps=ns;
        }
        printf("%u
    ",ans);
        
        return 0;
    }
  • 相关阅读:
    [原]OS X 10.9 Mavericks
    [原]iOS Makefile Template
    [原]Escape From the iOS Sanbox on Jailbreak Device
    串->串的表示和实现
    队列->队列的应用(银行业务模拟)
    队列->队列的表示和实现
    栈->栈与递归
    栈->栈的应用
    栈->栈的基本定义
    线性表->应用->一元多项式
  • 原文地址:https://www.cnblogs.com/AKCqhzdy/p/10679897.html
Copyright © 2011-2022 走看看