zoukankan      html  css  js  c++  java
  • 【二分】【预处理】zoj4029 Now Loading!!!

    题意:给定一个序列,多次询问

    将a数组从小到大排序,下面那个值只有不超过32种,于是预处理f[i][j],表示分母为i时,aj/i的前缀和是多少。

    然后对于一个给定的p,一定将分母划分成了一些连续的段落,通过枚举这些分母,二分获得分母变化的位置,将区间和累计进答案。

    注意,对于给定的p,一个分母控制的某段区间是a的元素属于p^i+1到p^(i+1)的这段。

    复杂度log^2n。

    #include<cstdio>
    //#include<cmath>
    #include<algorithm>
    using namespace std;
    int f,cc;
    void R(int &x){
        cc=0;f=1;
        for(;cc<'0'||cc>'9';cc=getchar())if(cc=='-')f=-1;
        for(x=0;cc>='0'&&cc<='9';cc=getchar())(x*=10)+=(cc-'0');
        x*=f;
    }
    #define MOD 1000000000
    typedef long long ll;
    int T,n,m,a[35][500005];
    ll c[500005];
    //int poses[100005][35];
    //double Log(const int &p,const int &x){
    //	return log((double)x)/log((double)p);
    //}
    int main(){
    	R(T);
    	for(;T;--T){
    		R(n); R(m);
    		for(int i=1;i<=n;++i){
    			R(a[1][i]);
    		}
    		sort(a[1]+1,a[1]+n+1);
    		for(int i=1;i<=n;++i){
    			c[i]=(ll)a[1][i];
    		}
    		for(int i=1;i<=n;++i){
    			for(int j=2;j<=32;++j){
    				a[j][i]=a[1][i]/j;
    			}
    		}
    		for(int i=1;i<=32;++i){
    			for(int j=1;j<=n;++j){
    				a[i][j]=(a[i][j]+a[i][j-1])%MOD;
    			}
    		}
    //		for(int p=2;p*p<=MOD;++p){
    //			for(int i=1;i<=32;++i){
    //				if((int)(ceil(Log(p,c[n]))+0.5)<i){
    //					poses[p][i]=n+1;
    //					continue;
    //				}
    //				int l=1,r=n;
    //				while(l<r){
    //					int mid=(l+r>>1);
    //					if((int)(ceil(Log(p,c[mid]))+0.5)>=i){
    //						r=mid;
    //					}
    //					else{
    //						l=mid+1;
    //					}
    //				}
    //				poses[p][i]=l;
    //			}
    //		}
    		int p;
    		ll ans=0;
    		int popo[35];
    		for(int zu=1;zu<=m;++zu){
    			R(p);
    //			if((ll)p*(ll)p<=(ll)MOD){
    //				for(int i=1;i<32;++i){
    //					if(poses[p][i]>n){
    //						break;
    //					}
    //					ans=(ans+((ll)zu*(ll)((a[i][poses[p][i+1]-1]-a[i][poses[p][i]-1]+MOD)%MOD))%(ll)MOD)%(ll)MOD;
    //				}
    //			}
    //			else{
    				ll pp=1;
    				for(int i=1;i<32;++i){
    					ll* be=upper_bound(c+1,c+n+1,pp);
    					if(be-c>n){
    						break;
    					}
    					pp*=(ll)p;
    					ans=(ans+((ll)zu*(ll)((a[i][upper_bound(c+1,c+n+1,pp)-c-1]-a[i][be-c-1]+MOD)%MOD))%(ll)MOD)%(ll)MOD;
    				}
    //			}
    		}
    		printf("%lld
    ",ans);
    	}
    	return 0;
    }
  • 相关阅读:
    团队展示&选题
    结对编程(JAVA实现)
    wc项目(node.js实现)
    复审与事后分析
    事后诸葛亮分析报告
    Alpha阶段项目复审
    测试与发布
    Scrum 冲刺第五篇
    Scrum 冲刺第一篇
    项目冲刺
  • 原文地址:https://www.cnblogs.com/autsky-jadek/p/9058402.html
Copyright © 2011-2022 走看看