莫比乌斯
今年的多校比赛,莫比乌斯反演的题目经常出现,但是我们队对于这种题可以说是直接放掉,不是因为没学过,多少了解一些,但是也只是皮毛,导致根本就做不出来,其实想一想,其实次数多了,就可以看出原因了,没有过总结,没有过思考,昨天心血来潮,好好的看了一下莫比乌斯反演,有一点感悟!
-
什么是莫比乌斯反演
...
可以通过 反向推导出
...
公式:
其中 即为莫比乌斯函数。
也可以如下定义:
-
分解质因数,无不相同,则
-
其他为0
性质:积性函数
莫比乌斯(容斥的优化)——我的理解
怎么求莫比乌斯函数?
-
类似于筛素数的方案,O(nlogn),一般足够了。
-
级别,无法预处理出1~n的莫比乌斯表,利用因式分解,讨论因子的拿法(类似压状),O(sqrt(n))
栗子:
//求莫比乌斯函数 O(nlogn)
void getMu() {
for(int i = 1; i < maxn; i++) {
int target = i == 1 ? 1 : 0;
int delta = target - mu[i];
mu[i] = delta;
for(int j = i + i ; j < maxn; j+=i)
mu[j] +=delta;
}
}
// n 的 约数的莫比乌斯函数值map形式返回 O(sqrt(n))
map<int,int> moebius(int n) {
map<int,int> res;
vector<int> primes;
//枚举n的质数
for(int i = 2; i*i <= n; i++) {
if(n%i==0) {
primes.push_back(i);
while(n%i==0) n/=i;
}
}
if(n!=1) primes.push_back(n);
int m = primes.size();
//不超过n的约数个数 次
for(int i = 0; i < (1<<m); i++) {
int mu = 1,d = 1;
for(int j = 0; j < m; j++) {
if(i>>j&1) {
mu*=-1;
d*=primes[j];
}
}
res[d] = mu;
}
return res;
}
例题一:
分析:总共26^n,根据容斥,枚举最短循环节d,减去26^d,根据莫比乌斯函数来容斥优化。注意数据范围极大!
例题二:
分析:
注意:(1,3),(3,1)相同。
先考虑不同的情况:
即:两个区间内互质的对数,同理:总共 对,减去不符合情况的对数,枚举gcd,就有 * 对,然后用莫比乌斯来容斥。
怎么解决(1,3)(3,1)相同的情况呢?
还是容斥!!!
减去较小区间做一遍上述操作的一半。
例题三:
题意:给定一个正方体的区间,从坐标(0,0,0)处可以看到多少个整点?
分析:(1,1,1) 能看到,但是(2,2,2)就三点共线挡住了,同理(2,3,4)能看到,但是(4,6,8)就看不到了,可以看出就是 的点,被除以gcd的点挡住了。也就是求 (x,y,z)互质的点对,就是上一题加一维。
值得注意的是区间是[1,n],整个坐标系可以x = 0等,x等于0,相当于一个平面,减一围。
还有3个坐标轴上的点。
到此,你就已经入门莫比乌斯了~~~
TreeDream