• bzoj4305: 数列的GCD


    要求k个与原序列中的数不同,就是要求(n-k)个相同,令K=n-k

    然后cnt[i]表示序列a中i的倍数的个数

    f[i]表示gcd为i的倍数的方案数

    f[i]=C(cnt[i],K)*(m/i-1)^(cnt[i]-K)*(m/i)^(n-cnt[i])

    那么ans[i]=f[i]-sigma(ans[j])  (j%i==0)

    cnt和组合数都可以在nlogn内预处理

    所以复杂度nlogn

    详见代码

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <cmath>
    #include <cstdlib>
    #include <algorithm>
    #define ll long long
    #define N 300005
    #define P 1000000007
    
    using namespace std;
    inline int read(){
    	int ret=0;char ch=getchar();
    	while (ch<'0'||ch>'9') ch=getchar();
    	while ('0'<=ch&&ch<='9'){
    		ret=ret*10-48+ch;
    		ch=getchar();
    	}
    	return ret;
    }
    
    int pow2(int x,int y){
    	int ret=1;
    	while (y){
    		if (y&1) ret=(ll)ret*x%P;
    		y=y>>1;
    		x=(ll)x*x%P;
    	}
    	return ret;
    }
    
    int n,m,K;
    int a[N];
    int cc[N];
    int cnt[N];
    void precompute(){
    	memset(cnt,0,sizeof(cnt));
    	for (int i=1;i<=n;++i) ++cnt[a[i]];
    	for (int i=1;i<=m;++i)
    		for (int j=2;i*j<=m;++j) cnt[i]+=cnt[i*j];
    	cc[K]=1;
    	for (int i=K+1;i<=n;++i) cc[i]=(ll)cc[i-1]*pow2(i-K,P-2)%P*i%P;
    }
    
    int ans[N];
    
    int main(){
    	n=read();m=read();K=n-read();
    	for (int i=1;i<=n;++i) a[i]=read();
    	precompute();
    	for (int i=m;i;--i)if (cnt[i]>=K){
    		ans[i]=(ll)cc[cnt[i]]*pow2(m/i-1,cnt[i]-K)%P*pow2(m/i,n-cnt[i])%P;
    		for (int j=2;i*j<=m;++j) (ans[i]-=ans[i*j]-P)%=P;
    	}
    	else ans[i]=0;
    	for (int i=1;i<m;++i) printf("%d ",ans[i]);
    	printf("%d
    ",ans[m]);
    	return 0;
    }
    

      

  • 相关阅读:
    sync.Once.Do(f func())
    协程
    Qt 线程基础(QThread、QtConcurrent、QThreadPool等)
    linux下valgrind的使用概述
    QT--QSocketNotifier类介绍
    QThreadPool类和QtConcurrent命名空间
    联想电池维修
    asm
    tapset::iosched(3)
    systemtap --diskio
  • 原文地址:https://www.cnblogs.com/wangyurzee7/p/5129656.html
Copyright © 2020-2023  润新知