Sky Code

Sky Code

给出n个数,求选出4个数组合,使其gcd为1,,\(n<=10000\),每个数\(<=10000\)

理解1:容斥原理

注意到Mobius反演式子不好写出,于是我们考虑它的兄弟,容斥,于是设\(F(d)\)表示数中有约数d的个数,所以由容斥原理,我们不难得到

\[ans=\sum_{d=1}^{10000}F(d)\mu(d)\]

预处理出函数\(\mu\),和\(F\),代入式子枚举即可。

理解2:Mobius反演

考虑到无法写出具体的式子,于是我们可以列出抽象式子,设\(f(d)\)为选出4个数gcd为d的个数,设\(F(d)\)表示为选出4个数公约数为d的方案数,所以由Mobius反演定理,我们有

\[f(d)=\sum_{d|x}F(x)\mu(x/d)\]

所以

\[ans=f(1)\sum_{x=1}^{10000}F(x)\mu(x)\]

代入式子枚举即可。

参考代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#define il inline
#define ri register
#define ll long long
using namespace std;
ll rc[10001];
bool check[10001];
int prime[2001],pt,mu[10001];
il ll C4(ll);
il void prepare(int);
int main(){
    int n,i,j;prepare(10000);ll ans;
    while(scanf("%d",&n)!=EOF){
        ans&=0,memset(rc,0,sizeof(rc));
        while(n--){
            scanf("%d",&i);
            for(j=1;j*j<i;++j)
                if(!(i%j))++rc[j],++rc[i/j];
            if(j*j==i)++rc[j];
        }for(i=1;i<=10000;++i)ans+=mu[i]*C4(rc[i]);
        printf("%lld\n",ans);
    }
    return 0;
}
il ll C4(ll n){
    return n*(n-1)*(n-2)*(n-3)/24;
}
il void prepare(int n){
    int i,j;mu[1]=1;
    for(i=2;i<=n;++i){
        if(!check[i])prime[++pt]=i,mu[i]=-1;
        for(j=1;j<=pt&&prime[j]*i<=n;++j){
            check[i*prime[j]]|=true;
            if(!(i%prime[j]))break;
            mu[i*prime[j]]=~mu[i]+1;
        }
    }
}

转载于:https://www.cnblogs.com/a1b3c7d9/p/10839611.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值