莫比乌斯反演学习笔记

莫比乌斯反演 学习笔记

因为在 NDPC 中发现自己并不会莫比乌斯反演于是来学习了

目录

积性函数

一说数论函数, 我个人认为积性函数这个叫法更好

对于一个函数 \(f(x)\), 如果满足对于任意的 $(a, b) | \(gcd(a, b) = 1, a \in \mathbb{Z}, b \in \mathbb{Z}\), \(f(ab) = f(a)f(b)\), 那么称这个函数为积性函数(数论函数)

特别地, 如果对于所有的 \((a, b) | a \in \mathbb{Z}, b \in \mathbb{Z}\) 都有 \(f(ab) = f(a)f(b)\) 的话, 函数 \(f\) 被称为 完全积性函数(完全数论函数)

二者区别在于是否要求两整数互质

常见的积性函数

  1. 常数函数: \(1(n) = 1\) (完全积性函数)
  2. 单位函数: \(\varepsilon(n) = n = 1\)[[1]](#[1]) (完全积性函数)
  3. 恒等函数: \(id_k(n) = n^k\), 这里通常所说的恒等函数当 \(k\) 取 \(1\), 即 \(id_1\), 一般记作 \(id\) (完全积性函数)
  4. 除数函数: \(\sigma_k(n) = \displaystyle\sum_{d | n} d^k\), 通常我们把 \(\sigma_1(n)\) 记作 \(\sigma(n)\) 或者 \(d(n)\), 实际意义是 \(n\) 的因数和
  5. 欧拉函数: \(\varphi(n) = \displaystyle\sum_{i = 1}^{n}gcd(i, n) = 1\), 即统计小于等于 \(n\) 的数中与 \(n\) 互质的数的个数
  6. 莫比乌斯函数: 简单说来: 当 \(n\) 等于 \(1\) 时, \(\mu(n) = 1\); 否则, 当 \(n\) 的素数分解中有非一次的素因子时 \(\mu(n) = 0\); 否则, 当 \(n\) 有奇数个素因子时, \(\mu(n) = -1\); 否则, \(\mu(n) = 1\). 形式化地

\\\mu(n) = \\begin{cases} 1, \& n = 1 \\\\ 0, \& n \\text{ 有平方素因子} \\\\ (-1)\^k, \& n = p_1 p_2 \\cdots p_k \\text{(}k\\text{ 个不同素因子)} \\end{cases} \\

推荐读者自己证明所有的积性函数的 积性

狄利克雷卷积

对于积性函数 \(f\) 和 \(g\), 它们的卷积 \(f \ast g\) 满足这样一个关系式:

\(f \\ast g)(n) = \\displaystyle\\sum_{d \| n}f(n)g(\\frac{n}{d}) \\

这里有一个良好的性质, 因为因数是成对出现的, 所以积性函数卷积显然满足交换律(当然卷积本身就满足交换律), 也就是说我们有

\(f \\ast g)(n) = \\displaystyle\\sum_{d \| n}f(n)g(\\frac{n}{d}) = \\displaystyle\\sum_{d \| n}g(n)f(\\frac{n}{d}) \\

莫比乌斯反演

莫比乌斯反演就是利用莫比乌斯函数和狄利克雷卷积将一些难以直接计算的量转化为便于计算的量从而降低复杂度

最常用的是下列恒等式

\\\displaystyle\\sum_{d \| n}\\mu(d) = \[n = 1 = \varepsilon(n) \]

Proof

令 \(n = \prod p_i^{e_i} (p \in primes)\)

我们再构造一个 \(n' = \prod p_i\)

因为当 \(e_i \ge 2\) 时 \(\mu(n) = 0\)

所以我们有 \(\displaystyle\sum_{d | n}\mu(d) = \displaystyle\sum_{d | n'}\mu(d)\)

此时根据二项式定理

\(\displaystyle\sum_{d | n'}\mu(d) = \displaystyle\sum_{i = 0}^{k}\binom{k}{i}(-1)^{i} = (1 + (-1)) ^ {i} = 0 ^ k\)

这个式子一般 等于 \(0\), 但当 \(k = 0\), 也就是不存在质因数 时, 这个式子等于 \(1\), 而一个正整数不存在质因数当且仅当它为 \(1\)

综上 \(\displaystyle\sum_{d | n}\mu(d) = n = 1 = \varepsilon(n)\)

感谢 OI-Wiki

根据狄利克雷卷积, 我们还可以知道 \(\varepsilon = 1 \ast \mu\)

以这一个最常见的莫比乌斯反演作为例子, 读者理解后应当可以推出大多其它的反演式子

求解莫比乌斯函数

根据定义我们显然可以枚举出来质因数然后 check, 是 \(O(\sqrt{n})\) 的, 但当我们要求出很多数的莫比乌斯函数的时候这个复杂度就有些难以接受了

为了解决这个问题, 我们来思考如何确定 \(n\) 的莫比乌斯函数的取值, 因为这是一个积性函数 , 且对于一个质数 \(p\), \(\mu(p)\) 一定等于 \(-1\). 所以当 \(n\) 为合数时, 我们可以枚举小于 \(n\) 的质数 \(p\), 然后分为两类:

  1. \(p | n\), 这个时候的 \(\mu(pn)\) 一定含有一个 \(p^2\) 的因子项, 所以 \(\mu(pn) = 0\)
  2. \(p \nmid n\), 这个时候 \(\mu(pn)\) 相当于在 \(n\) 的基础上添加了一个新的因子 \(p\), 由积性函数定义, \(\mu(pn) = \mu(p) \times \mu(n)\), 由我们刚才所讲的 \(\mu(p)\) 一定等于 \(-1\), 我们有 \(\mu(pn) = -1\mu(n) = -\mu(n)\)

我们发现这个过程很类似于筛法, 所以我们可以使用埃氏筛 \(O(n \ ln \ ln \ n)\) 的做

但事实上还可以优化, 如果我们使用线性筛, 则可以做到 \(O(n)\). 简要介绍一下线性筛的思想, 只用一个合数最小的质因子来筛除这个数, 具体可以看代码然后感性理解或者 OI-Wiki

而几乎在所有的本文提到的数论函数中, 线性筛其实是好写于埃氏筛的, 通常表现在于我们只需要找到一个最小的 \(n\) 的质因数\(p\), 这个目标是与线性筛相同的

这里贴一份代码

C++ 复制代码
// ========= Made By BilllIlllly =========
std::vector<bool> is_prime;
std::vector<int> mu;
std::vector<int> primes;

void sieve(int n) {
    is_prime.assign(n + 1, true);
    mu.assign(n + 1, 0);

    is_prime[1] = false;  // 如果你认为 1 是质数, 那你也可以去掉这一行
    mu[1] = 1;
    for (int i = 2; i <= n; ++i) {
        if (is_prime[i]) {
            primes.emplace_back(i);
            mu[i] = -1;
        }
        for (int p : primes) {
            if (i * p > n)
                break;
            is_prime[i * p] = false;
            if (i % p == 0) {
                mu[i * p] = 0;
                break;
            }
            mu[i * p] = -mu[i];
        }
    }
}

例题

互质数对个数

这是莫比乌斯反演最经(jian)典(dan)的一道例题, 虽然好像并没有找到被出成题的版本

给定一个正整数 \(n\), 求 \(\displaystyle\sum_{i = 1}^{n}\displaystyle\sum_{j = 1}^{n} gcd(i, j) = 1\), 简单来说就是在 \(1, n\) 中有多少个互质的整数对

朴素做法

枚举 \(i, j\), 然后求gcd, 是 \(O(n^2 log n)\) 的, 不能接受

反演做法

我们发现 \(gcd(i, j) = 1\) 可以写成 \(\varepsilon(gcd(i, j))\), 所以原式等于

\\\displaystyle\\sum_{i = 1}\^{n}\\displaystyle\\sum_{j = 1}\^{n} \\varepsilon(gcd(i, j)) \\

由莫比乌斯反演, 原式等于

\\\displaystyle\\sum_{i = 1}\^{n}\\displaystyle\\sum_{j = 1}\^{n}\\displaystyle\\sum_{d \| gcd(i, j)} \\mu(i) \\

\= \\displaystyle\\sum_{i = 1}\^{n}\\displaystyle\\sum_{j = 1}\^{n}\\displaystyle\\sum_{\\substack{d \|i \\\\ d \| j }} \\mu(i) \\

考虑从枚举 \((i, j)\) 变为枚举 \(d\), 原式等于

\\\displaystyle\\sum_{d = 1}\^n \\sum_{d \| i}\^{n}\\sum_{d \| j}\^{n} \\mu(d) \\

这里需要注意的是, 虽然我们写的是 \(d | i, d | j\), 但事实上这两个求和的求和记号分别是 \(i, j\), 简单来说我们实际上实在令 \(i, j\) 是 \(d\) 的倍数

这个时候发现原式的 \(\mu(d)\) 和 \(i, j\) 无关, 所以我们直接交换求和次序, 原式等于

\\\displaystyle\\sum_{d = 1}\^n \\mu(d) \\sum_{d \| i}\^{n}\\sum_{d \| j}\^{n} 1 \\

最后我们考虑这里的后面两个 \(\sum\) 怎么处理, 事实上, \(i, j\) 的取值数量为 \(n\) 中的 \(d\) 的倍数的数量, 也就是 \(\lfloor \frac{n}{d} \rfloor\), 所以总共有 \(\lfloor \frac{n}{d} \rfloor ^ 2\) 个取值, 原式就终于被我们化为了

\\\displaystyle\\sum_{d = 1}\^n (\\mu(d) \\lfloor \\frac{n}{d} \\rfloor \^ 2) \\

我们分析这个式子

  • \(\mu(d)\) 事实上是可以通过预处理达到 \(O(1)\) 的, 预处理的复杂度最常见的可以线性筛(前文讲过); 或者如果需求更快, 可以用杜教筛; 想要更快, 可以用洲阁筛或者Min-25筛(当然我只会线性筛).
  • \(\lfloor \frac{n}{d} \rfloor ^ 2\) 显然可以 \(O(1)\) 每次的求出

综上, 通过线性筛预处理, 我们 至少 可以做到 \(O(n)\).

优化:整除分块

现在具体再来讲一下莫比乌斯反演最严厉的父亲------整除分块。

预处理出 \(\mu\) 的前缀和,利用整除分块 (也叫数论分块),我们可以把单次查询的复杂度降到 \(O(\sqrt{n})\)!

观察式子 \(\lfloor \frac{n}{d} \rfloor\),当 \(d\) 在 \(1, n\) 内变化时,这个式子的取值其实只有 \(O(\sqrt{n})\) 种。

为什么呢?

  • 当 \(d \le \sqrt{n}\) 时,\(\lfloor \frac{n}{d} \rfloor\) 显然最多只有 \(\sqrt{n}\) 种取值。
  • 当 \(d > \sqrt{n}\) 时,\(\lfloor \frac{n}{d} \rfloor \le \sqrt{n}\),所以也最多只有 \(\sqrt{n}\) 种取值。

综上,取值总数不超过 \(2\sqrt{n}\)。而且,对于同一个取值 \(k = \lfloor \frac{n}{d} \rfloor\),满足条件的 \(d\) 一定是一段连续的区间 \(l, r\)。

这个区间的右端点 \(r\) 怎么求呢?

由 \(k = \lfloor \frac{n}{d} \rfloor \le \frac{n}{d}\),可得 \(d \le \frac{n}{k}\),所以最大的 \(d\) 就是 \(\lfloor \frac{n}{k} \rfloor\)。

也就是说,右端点 \(r = \lfloor \frac{n}{\lfloor n/l \rfloor} \rfloor\)。

这样,我们就可以把 \(d\) 分成 \(O(\sqrt{n})\) 段,每一段内的 \(\lfloor \frac{n}{d} \rfloor^2\) 是相同的,提出来之后,只需要乘上这一段 \(\mu(d)\) 的和(即前缀和之差)即可。

预处理前缀和 \(O(n)\),单次查询 \(O(\sqrt{n})\)。我们将一些问题的复杂度从最初的 \(O(n^2 log n)\) 降为了 \(O(\sqrt{n})\)!

YY的GCD

有了整除分块,我们来看一道真正的板子题。

题目大意 :给定 \(N, M\),求 \(\displaystyle\sum_{i=1}^N \sum_{j=1}^M \\gcd(i,j) \\in \\text{prime}\)。多组询问。

推导过程

首先,枚举质数 \(p\):

\\\sum_{p \\in \\text{prime}} \\sum_{i=1}\^N \\sum_{j=1}\^M \[\\gcd(i,j) = p \]

把 \(p\) 除进去:

\= \\sum_{p \\in \\text{prime}} \\sum_{i=1}\^{\\lfloor N/p \\rfloor} \\sum_{j=1}\^{\\lfloor M/p \\rfloor} \[\\gcd(i,j) = 1 \]

使用基本反演公式 \(\\gcd(i,j)=1 = \sum_{d|\gcd(i,j)} \mu(d)\):

\= \\sum_{p \\in \\text{prime}} \\sum_{i=1}\^{\\lfloor N/p \\rfloor} \\sum_{j=1}\^{\\lfloor M/p \\rfloor} \\sum_{d\|\\gcd(i,j)} \\mu(d) \\

把 \(d\) 提出来先枚举:

\= \\sum_{p \\in \\text{prime}} \\sum_{d=1}\^{\\min(\\lfloor N/p \\rfloor, \\lfloor M/p \\rfloor)} \\mu(d) \\lfloor \\frac{N}{pd} \\rfloor \\lfloor \\frac{M}{pd} \\rfloor \\

这里令 \(T = pd\),则 \(d = T/p\)。我们改变枚举顺序,先枚举 \(T\):

\= \\sum_{T=1}\^{\\min(N,M)} \\lfloor \\frac{N}{T} \\rfloor \\lfloor \\frac{M}{T} \\rfloor \\sum_{p\|T, p \\in \\text{prime}} \\mu\\left(\\frac{T}{p}\\right) \\

令 \(f(T) = \displaystyle\sum_{p|T, p \in \text{prime}} \mu\left(\frac{T}{p}\right)\)。

那么原式 = \(\displaystyle\sum_{T=1}^{\min(N,M)} \lfloor \frac{N}{T} \rfloor \lfloor \frac{M}{T} \rfloor f(T)\)

对于 \(f(T)\),我们可以用类似埃氏筛的方法在 \(O(N \log N)\) 内预处理出来,并求出前缀和 \(S(T) = \sum_{i=1}^T f(i)\)。

cpp 复制代码
// 预处理 f(T)
for (int i = 1; i <= MAXN; ++i) {
    if (is_prime[i]) {
        for (int j = 1; i * j <= MAXN; ++j) {
            f[i * j] += mu[j];
        }
    }
}
// 求前缀和
for (int i = 1; i <= MAXN; ++i) {
    sum_f[i] = sum_f[i - 1] + f[i];
}

然后对于每次询问,利用整除分块,可以在 \(O(\sqrt{N} + \sqrt{M})\) 的时间内求出答案:

cpp 复制代码
long long solve(int n, int m) {
    long long res = 0;
    int limit = min(n, m);
    for (int l = 1, r; l <= limit; l = r + 1) {
        r = min(n / (n / l), m / (m / l));
        res += (long long)(n / l) * (m / l) * (sum_f[r] - sum_f[l - 1]);
    }
    return res;
}

预处理 \(O(N \log N)\),单次询问 \(O(\sqrt{N})\),这题就愉快地解决了。

NDPC_L - LCM

现在回顾这道让我学这个东西的题

题意

给定 \(N\) 和 \(1 \dots N\) 的一个排列 \(A\)。

有一个 \(N\) 个顶点的有向图,对于 \(1 \le i < j \le N\),从顶点 \(i\) 到顶点 \(j\) 有 \(\text{lcm}(A_i, A_j)\) 条有向边。

求对于每个 \(v = 2, 3, \dots, N\),从顶点 1 到顶点 \(v\) 的路径数,对 998244353 取模。两条路径不同当且仅当它们经过的边的集合不同。

\(N \le 2 \times 10^5\)。

做法

首先,最直接的思路肯定是 DP(废话这都是ndpc了)。

设 \(dp_j\) 为从顶点 1 到顶点 \(j\) 的路径数。显然 \(dp_1 = 1\)。

对于 \(j > 1\),转移方程为:

\dp_j = \\sum_{i=1}\^{j-1} dp_i \\times \\text{lcm}(A_i, A_j) \\

这坨直接算是 \(O(N^2)\) 的,对于 \(N = 2 \times 10^5\) 显然你 T 飞了。考虑优化。

看到 \(\text{lcm}\),显然要把它转化为 \(\gcd\):

\\\text{lcm}(x, y) = \\frac{x y}{\\gcd(x, y)} \\

代入转移方程:

\dp_j = \\sum_{i=1}\^{j-1} dp_i \\frac{A_i A_j}{\\gcd(A_i, A_j)} = A_j \\sum_{i=1}\^{j-1} dp_i A_i \\frac{1}{\\gcd(A_i, A_j)} \\

现在问题变成了如何快速处理 \(\frac{1}{\gcd(x, y)}\)。

我们知道,对于任意积性函数 \(f\),都可以写成 \(f(\gcd(x, y)) = \sum_{d|\gcd(x,y)} g(d)\) 的形式,其中 \(g = f \ast \mu\)。

这里 \(f(n) = \frac{1}{n}\),我们来求一下对应的 \(g(d)\)(为了方便,下面记作 \(h(d)\)):

\h(d) = \\sum_{k\|d} f(k) \\mu\\left(\\frac{d}{k}\\right) = \\sum_{k\|d} \\frac{1}{k} \\mu\\left(\\frac{d}{k}\\right) \\

因为 \(f\) 和 \(\mu\) 都是积性函数,所以 \(h\) 也是积性函数。我们只需要算它在质数幂 \(p^a\) 处的值:

\h(p\^a) = \\sum_{i=0}\^a \\frac{1}{p\^i} \\mu(p\^{a-i}) \\

因为 \(\mu(p^{a-i})\) 只有在 \(a-i = 0\) 或 \(1\) 时非零,所以只有最后两项:

\h(p\^a) = \\frac{1}{p\^a}\\mu(1) + \\frac{1}{p\^{a-1}}\\mu(p) = \\frac{1}{p\^a} - \\frac{1}{p\^{a-1}} = \\frac{1-p}{p\^a} \\

所以对于任意 \(n = \prod p_i^{a_i}\):

\h(n) = \\prod \\frac{1-p_i}{p_i\^{a_i}} = \\frac{1}{n} \\prod_{p\|n} (1-p) \\

于是我们得到了一个非常漂亮的恒等式:

\\\frac{1}{\\gcd(x, y)} = \\sum_{d\|\\gcd(x,y)} h(d) \\

将这个恒等式代回 \(dp_j\) 的式子中:

\dp_j = A_j \\sum_{i=1}\^{j-1} dp_i A_i \\sum_{d\|\\gcd(A_i, A_j)} h(d) \\

交换求和顺序,先枚举 \(d\)。因为 \(d\) 必须同时整除 \(A_i\) 和 \(A_j\),所以 \(d\) 必须是 \(A_j\) 的约数:

\dp_j = A_j \\sum_{d\|A_j} h(d) \\sum_{i=1}\^{j-1} dp_i A_i \[d\|A_i \]

令 \(S_d = \sum_{i=1}^{j-1} dp_i A_i d\|A_i\),表示前面所有满足 \(d|A_i\) 的 \(dp_i A_i\) 的和。

那么转移方程就变成了:

\dp_j = A_j \\sum_{d\|A_j} h(d) S_d \\

在计算出 \(dp_j\) 之后,我们需要更新所有的 \(S_d\)。对于 \(A_j\) 的每一个约数 \(d\),我们将 \(dp_j A_j\) 加到 \(S_d\) 中:

\S_d \\leftarrow S_d + dp_j A_j \\

复杂度

这个算法的复杂度如何呢?

对于每个 \(j\),我们需要枚举 \(A_j\) 的所有约数。

因为 \(A\) 是 \(1 \dots N\) 的排列,所以 \(A_j \le N\)。

一个数 \(x \le N\) 的约数个数平均是 \(O(\log N)\)。

所以枚举约数的总复杂度是 \(\sum_{j=1}^N d(A_j) = \sum_{x=1}^N d(x) = O(N \log N)\)。

预处理 \(h(d)\) 和每个数的列表,也显然可以在 \(O(N \log N)\) 的复杂度内完成

所以总时间复杂度 \(O(N \log N)\),空间复杂度 \(O(N \log N)\)

代码实现

这里贴一份核心代码:

C++ 复制代码
// divs[x] 存储了 x 的所有约数
// h[d] 就是 h(d)
// S[d] 维护 S_d

static constexpr int MOD = 998244353

std::vector<long long> dp(N + 1, 0);
std::vector<long long> S(N + 1, 0);
dp[1] = 1;

// 初始化 S,把 dp[1] * A[1] 加到 A[1] 的所有约数对应的 S 中
long long val1 = A[1]; // dp[1] * A[1]
for (int d : divs[A[1]]) S[d] = (S[d] + val1) % MOD;

for (int j = 2; j <= N; ++j) {
    long long sum = 0;
    // 计算 dp[j]
    for (int d : divs[A[j]]) sum = (sum + S[d] % MOD * h[d]) % MOD; 
    dp[j] = sum * A[j] % MOD;
    
    // 输出答案
    std::cout << dp[j] << '\n';
    
    // 更新 S,为了后面的 j 转移
    if (j < N) {
        long long val = dp[j] * A[j] % MOD;
        for (int d : divs[A[j]]) S[d] = (S[d] + val) % MOD;
    }
}

代码里的 h[d] 可以提前用 YY的GCD 的方法预用线性筛处理出来,初始时 h[i] = inv[i],然后对于每个质数 p,把它的倍数的 h 都乘上 (1 - p + MOD) % MOD 即可。


更进一步!

这里往下会放我遇到的有代表性的莫比乌斯反演问题

咕咕咕......


  1. 这里的\[\]是艾弗森记号, 当里面的内容成立时它的值为 \(1\), 否则其为 \(0\) ↩︎
相关推荐
stolentime2 小时前
CF2066D1 Club of Young Aircraft Builders (easy version)题解
c++·算法·动态规划·组合数学
Dillon Dong2 小时前
【风电控制】高低穿现场失败的原因分析——算法简单但工程复杂
算法·变流器·风电控制·dfig
小欣加油2 小时前
leetcode41 缺失的第一个正数
数据结构·c++·算法·leetcode
I Promise342 小时前
智驾APA_HPA可行驶区域检测算法工程师面试问题整理可参考
算法·面试·职场和发展
智者知已应修善业2 小时前
【51单片机按键控制1分钟正计时倒计时暂停复位】2024-1-2
c++·经验分享·笔记·算法·51单片机
weixin_468466852 小时前
UNet 模型结构从零搭建与实战解析
人工智能·深度学习·算法·机器学习·ai·unet
Useasy_JIJIANYUN3 小时前
合作快讯:极简云呼叫中心(Useasy)正式上架Zoho全球应用市场!
算法
isyoungboy3 小时前
Delaunay 拓扑图割法一种特征抽稀算法
算法
Shan12053 小时前
算法案例精讲:连接所有点的最小费用
算法