容斥与简单莫反

容斥与莫比乌斯函数

容斥原理:

介绍:设集合\(S_1\sim S_n\),记\(|S_i|\)表示集合\(S_i\)的大小,设\(\cup\)表示集合的并集运算,\(\cap\)表示集合的交集运算,则

\[\left|\bigcup_{i=1}^nS_i\right|=\sum_{i=1}^n|S_i|-\sum_{1\le i<j\le n}|S_i\cap S_j|+\sum_{1\le i<j<k\le n}|S_i\cap S_j\cap S_k|-…+(-1)^{n+1} \left | \bigcap_{i=1}^nS_i \right| \]

所以由容斥原理可以得到之前的多重集的组合数公式:

设集合\(\mathbb{S}={\lbrace n_1·a_1,n_2·a_2,n_3·a_3…n_k·a_k\rbrace}\)是由\(n_1\)\(a_1\)……组成的多重集,\(n=\sum_{i=1}^kn_i\)
则从该集合中选出\(r(r\le n)\)个数组成的不同的多重集数量为:

\[C_{k+r-1}^{k-1}-\sum_{i=1}^kC^{k-1}_ {k+r-n_i-2}+\sum_{1\le i<j \le n}C_{k+r-n_i-n_j-2}^{k-1}-…+(-1)^kC_{k+r-n-k-1}^{k-1} \]

对于容斥原理在代码中的实现:我们可以枚举\(x\in [0,2^k-1]\),将x内的含1的位提取出来,根据奇偶性判断是加还是减,特别的,我们令x=0代表\(C_{k+r-1}^{k-1}\)

Mobius函数

莫比乌斯函数是一个容斥系数,与容斥原理息息相关

定义:设\(x\)被算术基本定理分解为:\(\prod_{i=1}^np_i^{c_i}\)

\[\mu(x) = \begin{cases} 0 & \exists i\in[1,n],c_i\ge 2\\ 1 & n \bmod 2=0,\forall i\in[1,n],c_i=1\\ -1 & n \bmod 2=1,\forall i\in[1,n],c_i=1\\ \end{cases} \]

则称\(\mu(x)\)\(u\)\(mobius\)函数

对于mobius函数的求法:

如果只是求一个数的莫比乌斯函数,分解质因数即可,若是求\(1\sim N\)的,则使用埃拉托尼斯筛法计算

for(int i=1;i<=n;i++)mui[i]=1,vis[i]=0;
for(int i=2;i<=n;i++){  
    if(vis[i])continue;
    mui[i]=-1;
    vis[i]=1;
    for(int j=2;j*i<=n;j++){  
        mui[i*j]*=-1;
        if(j%i==0)mui[i*j]=0;
        vis[i*j]=1;
    }
}

性质:

1.对于任意正整数n,\(\sum_{d|n}\mu(i)=[n=1]\)

2.若\(\gcd(a,b)=1\),则\(\mu(ab)=\mu(a)\mu(b)\)

\[\sum_{d|n}\frac{\mu(d)}{d}=\frac{\varphi(n)}{n} \]

莫比乌斯函数与容斥原理的关系:

即对\(1\sim N\)应用容斥原理,则每个数的莫比乌斯函数决定了那个数的系数

随意扔一道例题:
ZAP-Queries

[POI2007]ZAP-Queries

题目描述

密码学家正在尝试破解一种叫 BSA 的密码。

他发现,在破解一条消息的同时,他还需要回答这样一种问题:

给出 \(a,b,d\),求满足 \(1 \leq x \leq a\)\(1 \leq y \leq b\),且 \(\gcd(x,y)=d\) 的二元组 \((x,y)\) 的数量。

因为要解决的问题实在太多了,他便过来寻求你的帮助。

输入格式

输入第一行一个整数 \(n\),代表要回答的问题个数。

接下来 \(n\) 行,每行三个整数 \(a,b,d\)

输出格式

对于每组询问,输出一个整数代表答案。

数据规模与约定

对于全部的测试点,保证 \(1 \leq n \leq 5 \times 10^4\)\(1 \leq d \leq a,b \leq 5 \times 10^4\)

分析:
求有多少对二元组\((x,y)\)满足\((x\le a,y \le b)\)并且\(gcd(x,y)=k\),等价于求有多少对二元组\((x,y)\)满足\((x\le \frac{a}{k},y \le \frac{b}{k})\)并且\(x,y\)互质
\(D[a,b,k]\)表示满足\(x\le a,y\le b\)并且\(k|gcd(x,y)\)的二元组有多少对,显然,只需要\(x,y\)\(k\)的倍数即可,而\(1\sim a\)中k的倍数有\(\lfloor\frac{a}{k}\rfloor\)个,b同理,则我们很容易就可以得出\(D[a,b,k]=\lfloor\frac{a}{k}\rfloor\lfloor\frac{b}{k}\rfloor\)

\(F[a,b]\)表示满足\(x\le a,y le b\)的二元组\((x,y)\)有多少对,则有:

\[F[a,b]=\sum_{i=1}^{\min(a,b)}\mu(i)\times D[a,b,i] \]

上式含义:当\(i=1\)的时候\(D[a,b,1]\times \mu(1)\),而\(\mu(1)=1\),所以此时就相当于是加上了所有的二元组\(a\times b\),应当减去\(2,3,5,7……\)的倍数,但是6,10等就被减去了两次需要在加回来一次……以此类推便可以确定对于每个数加减的系数都是\(\mu\)函数

又联系我们之前学习的数论分块的知识
\(\lfloor\frac{a}{k}\rfloor\lfloor\frac{b}{k}\rfloor\)这是呈段状的

\(\forall i\in [x,\min(\lfloor\frac{a}{\lfloor\frac{a}{x}\rfloor}\rfloor,\lfloor\frac{b}{\lfloor\frac{b}{x}\rfloor}\rfloor)],\lfloor\frac{a}{k}\rfloor\lfloor\frac{b}{k}\rfloor\)的值都相等,我们预处理\(\mu\)函数的前缀和即可直接累加这一段答案

#define int long long
int T,n,a,b,d,mobius[50005],sum[50005];
bool vis[50005];
void init(){
    for(int i=1;i<=50000;i++)mobius[i]=1;
    for(int i=2;i<=50000;i++){
    	if(!vis[i]){
    		mobius[i]=-1;
    		for(int j=2;j*i<=50000;j++){
    			vis[j*i]=1;
    			mobius[i*j]= j%i==0?0:-mobius[i*j];
			}
		}
	}
    for(int i=1;i<=50000;i++)sum[i]=sum[i-1]+mobius[i];
}
int get(int a,int x){
    return a/(a/x);
}
signed main(){
    init();
	  scanf("%lld",&T);
    while(T--){
        int ans=0;
        scanf("%lld%lld%lld",&a,&b,&d);
        a/=d,b/=d;
        n=min(a,b);
        for(int l=1,r;l<=n;l=r+1){
            r=min(n,min(get(a,l),get(b,l)));
            ans+=(sum[r]-sum[l-1])*(a/l)*(b/l);
        }
        printf("%lld\n",ans);
    }
    return 0;
}

莫比乌斯反演定理

仅作了解,tg无需掌握

设函数\(f(x),g(x)\)是定义在正整数集上的两个函数

形式1

若函数\(f(x),g(x)\)满足

\[f(n)=\sum_{d|n}g(d)=\sum_{d|n}g(\frac{n}{d}) \]

则:

\[g(n)=\sum_{d|n}\mu(d)f(\frac{n}{d})=\sum_{d|n}\mu(\frac{n}{d})f(d) \]

形式2

若函数\(f(x),g(x)\)满足:

\[f(n)=\sum_{n|d}g(d)=\sum_{n|d}g(\frac{n}{d}) \]

则有

\[g(n)=\sum_{n|d}\mu(\frac{d}{n})f(d)=\sum_{n|d}\mu(d)f(\frac{d}{n}) \]

推一个大佬博客“浅谈”莫反

posted @ 2022-11-30 22:23  spdarkle  阅读(34)  评论(0编辑  收藏  举报