莫比乌斯函数学习笔记

news/2025/1/19 11:48:29/文章来源:https://www.cnblogs.com/Rock-N-Roll/p/18679459

\(\text{莫比乌斯函数学习笔记}\)

前言

注意到我并没有将 "莫比乌斯反演" 作为文章的题目,主要原因是莫比乌斯反演可以解决的题目用莫比乌斯函数通常便可以解决。莫比乌斯函数的前置知识有且仅有数论分块。

数论分块

引理 1

\[\forall a,b,c\in\mathbb{Z},\left\lfloor\frac{a}{bc}\right\rfloor=\left\lfloor\frac{\left\lfloor\frac{a}{b}\right\rfloor}{c}\right\rfloor \]

证明:设 \(\dfrac a b=\lfloor \dfrac a b\rfloor + r(0\le r < 1)\),那么

\[\lfloor \frac{a}{bc}\rfloor=\lfloor \frac{a}{b}\cdot \frac{1}{c}\rfloor=\lfloor \frac{\lfloor \frac a b\rfloor}{c}+\frac r c\rfloor=\lfloor \frac{\lfloor \frac a b\rfloor}{c}\rfloor \]

引理 2

\[\forall n \in \mathbb{N}_{+}, \left|\left\{ \lfloor \frac{n}{d} \rfloor \mid d \in \mathbb{N}_{+},d\leq n \right\}\right| \leq \lfloor 2\sqrt{n} \rfloor \]

证明:

对于 \(d\le \lfloor \sqrt n\rfloor\)\(\lfloor\dfrac{n}{d}\rfloor\)\(\lfloor \sqrt n\rfloor\) 种取值;

对于 \(d>\lfloor \sqrt n\rfloor\)\(\lfloor \dfrac n d \rfloor\le \lfloor\sqrt n\rfloor\),也只有 \(\lfloor \sqrt n\rfloor\) 种取值。

结论

对于常数 \(n\),使得式子 \(\lfloor \dfrac{n}{i}\rfloor=\lfloor \dfrac n j\rfloor\) 成立且有 \(i\le j \le n\)\(j\) 值最大为 \(\left\lfloor\dfrac n{\lfloor\frac ni\rfloor}\right\rfloor\),这个值也就是 \(\lfloor \dfrac ni\rfloor\) 所在块的右端点。

应用

快速计算 \(\sum_{i=1}^nf(i)g(\lfloor\dfrac{n}{i}\rfloor)\) 的值。

我们从 \(l=1\) 开始枚举,每次考虑区间 \([l,\left\lfloor\dfrac n{\lfloor\frac ni\rfloor}\right\rfloor]\) 的贡献,即 \(g(\lfloor\dfrac nl\rfloor)\sum_{i-l}^{\lfloor\dfrac n{\lfloor\frac ni\rfloor}\rfloor}f(i)\) 的贡献。\(f\) 的贡献是容易的,\(g\) 的贡献可以直接计算。算完后令 \(l=\left\lfloor\dfrac n{\lfloor\frac ni\rfloor}\right\rfloor+1\),作为下一个枚举区间的左端点。那么时间复杂度显然是 \(O(\sqrt n)\)

代码:

int l = 1, r = 0;
while (l <= n) {r = n / (n / l);ans += (sum[r] - sum[l - 1]) * g[n / l];r = l + 1;
}

推广

若出现 \(\lfloor \dfrac {a_1}i\rfloor,\lfloor \dfrac {a_2}i\rfloor,\cdots,\lfloor \dfrac {a_n}i\rfloor\) 的多个整除式,要在每个区间的右端点中取最小值。

莫比乌斯函数

\(\mu\) 为莫比乌斯函数,定义为:

\[\mu(n)= \begin{cases} 1&n=1\\ 0&n\text{ 含有平方因子}\\ (-1)^k&k\text{ 为 }n\text{ 的本质不同质因子个数}\\ \end{cases} \]

需要知道的是 \(\mu\) 有以下性质:

性质 1

\(\mu\) 是积性函数,可以直接用欧拉筛求解。

性质 2

\[\sum_{d|n}\mu(d)= \begin{cases} 1&n=1\\ 0&n\neq 1\end{cases} \]

证明:设 \(n=\prod_{i=0}^k p_i^{c_i},n'=\prod_{i=1}^k p_i\),那么 \(\sum_{d|n}\mu(d)=\sum_{d|n'}\mu(d)=\sum_{i=0}^k{k\choose i}\cdot(-1)^i=(1+(-1))^k\)

由二项式定理,该式子的值只有在 \(k=0\),即 \(n=1\) 时为 \(1\),否则为 \(0\)

性质 3

\([\gcd(i,j)=1]\to \sum_{d|\gcd(i,j)}\mu(d)\)

这个结论比较重要,由 性质 2 是容易得出的。

那么其实莫比乌斯函数的部分就讲完了,让我们看几道例题。

例题

例 1:YY的GCD:求 \(\sum_{i=1}^n\sum_{j=1}^m[\gcd(i,j)\in\mathcal{P}]\)\(\mathcal{P}\) 为全体质数构成的集合。

不妨令 \(n<m\)

\[\begin{aligned} &\sum_{i=1}^n\sum_{j=1}^m[\gcd(i,j)\in\mathcal{P}]\\ =&\sum_{i=1}^n\sum_{j=1}^m\sum_{p\in\mathcal{P}}[\gcd(\frac ip,\frac jp)=1]\\ =&\sum_{p\in\mathcal{P}}\sum_{i=1}^{\lfloor\frac np\rfloor}\sum_{j=1}^{\lfloor \frac mp\rfloor}[\gcd(i,j)=1]\\ =&\sum_{p\in\mathcal{P}}\sum_{i=1}^{\lfloor\frac np\rfloor}\sum_{j=1}^{\lfloor \frac mp\rfloor}\sum_{d|\gcd(i,j)}\mu(d)\\ =&\sum_{p\in\mathcal{P}}\sum_{i=1}^{\lfloor\frac np\rfloor}\sum_{j=1}^{\lfloor \frac mp\rfloor}\sum_{d=1}^{\lfloor \frac np\rfloor}\mu(d)[d|i][d|j]\\ =&\sum_{p\in\mathcal{P}}\sum_{d=1}^{\lfloor \frac np\rfloor}\mu(d) \sum_{i=1}^{\lfloor\frac np\rfloor}[d|i]\sum_{j=1}^{\lfloor \frac mp\rfloor}[d|j]\\ =&\sum_{p\in\mathcal{P}}\sum_{d=1}^{\lfloor \frac np\rfloor}\mu(d)\lfloor \frac n{pd}\rfloor\lfloor\frac m{pd}\rfloor\\ =&\sum_{p\in \mathcal{P}}\sum_{T=1}^{\lfloor \frac np\rfloor}\mu(\frac Tp) \lfloor \frac nT\rfloor\lfloor \frac mT\rfloor\\ =&\sum_{T=1}^n\lfloor \frac nT\rfloor\lfloor \frac mT\rfloor\sum_{p\in\mathcal{P}\wedge p|T}\mu(\frac Tp) \end{aligned} \]

那么数论分块求解即可。

代码:

#include <bits/stdc++.h>
#define int long long
#define N 10000005
using namespace std;
vector<int>pri;
bool prm[N];
int mu[N];
int sm[N];
void pme(int n) {mu[1] = 1;for (int i = 2; i <= n; i++) {if (!prm[i]) pri.push_back(i), mu[i] = -1;for (auto j : pri) {if (i * j > n) break;prm[i * j] = 1;if (i % j == 0) {mu[i * j] = 0;break;}mu[i * j] = -mu[i];}}for (auto i : pri)for (int j = 1; j <= n / i; j++) sm[i * j] += mu[j];for (int i = 1; i <= n; i++) sm[i] += sm[i - 1];
}signed main() {ios::sync_with_stdio(0);cin.tie(0);pme(1e7);int T;cin >> T;while (T--) {int n, m;cin >> n >> m;if (n > m) swap(n, m);int l, r, ans = 0;for (l = 1; l <= n; l = r + 1) {r = min(n / (n / l), m / (m / l));ans += (sm[r] - sm[l - 1]) * (n / l) * (m / l);}cout << ans << '\n';}return 0;
}

例 2:[SDOI2017]数字表格

题意:第 \(i\) 行第 \(j\) 列的格子中的数是 \(f_{\gcd(i,j)}\),其中 \(\gcd(i,j)\) 表示 \(i,j\) 的最大公约数,,Doris 的表格中共有 \(n\times m\) 个数,她想知道这些数的乘积是多少。发现答案式是乘积,于是考虑将乘积转化为指数求和的形式。

\[\begin{aligned} &\prod_{i=1}^n\prod_{j=1}^m f(\gcd(i,j))\\ =&\prod_{p=1}^nf(p)^{\sum_{i=1}^n\sum_{j=1}^m[\gcd(i,j)=p]}\\ =&\prod_{p=1}^nf(p)^{\sum_{i=1}^{\lfloor\frac np\rfloor}\sum_{j=1}^{\lfloor \frac mp\rfloor}[\gcd(i,j)=1]}\\ =&\prod_{p=1}^nf(p)^{\sum_{i=1}^{\lfloor \frac np\rfloor}\sum_{j=1}^{\lfloor \frac mp\rfloor}\sum_{d=1}^{\lfloor \frac np\rfloor}\mu(d)\lfloor \frac n{pd}\rfloor\lfloor\frac m{pd}\rfloor}\\ =&\prod_{T=1}^n\prod_{p|T}f(p)^{\mu(\frac Tp)\lfloor \frac nT\rfloor\lfloor\frac mT\rfloor} \end{aligned} \]

那么这个东西看上去没怎么见过,实际上就是将数论分块的过程由加和改为累乘,维护的时候拿逆元和快速幂处理一下即可。注意处理 \(\mu(d)<0\) 时要拿扩展欧拉定理处理一下。

代码:

#include <bits/stdc++.h>
#define N 1000000
#define M (N + 5)
#define int long long
#define mod 1000000007
using namespace std;
int qpow(int x, int y) {if (y < 0) y = (y + mod - 1) % (mod - 1);int ans = 1;while (y) {if (y & 1) ans = ans * x % mod;x = x * x % mod;y >>= 1;}return ans;
}
int n, m;
int f[M];
void gtf() {f[1] = 1;for (int i = 2; i <= N; i++)f[i] = (f[i - 1] + f[i - 2]) % mod;
}
vector<int>prm;
bool is[M];
int mu[M];int sm[M], fac[M], inv[M];
void solve() {mu[1] = 1;for (int i = 2; i <= N; i++) {if (!is[i]) {prm.push_back(i);mu[i] = -1;} for (auto j : prm) {if (i * j > N) break;is[i * j] = 1;if (i % j == 0) {mu[i * j] = 0;break;}mu[i * j] = -mu[i];}}for (int i = 0; i <= N; i++) sm[i] = 1;for (int i = 1; i <= N; i++)for (int j = 1; j <= N / i; j++)sm[i * j] = sm[i * j] * qpow(f[i], mu[j]) % mod;fac[0] = 1;for (int i = 1; i <= N; i++) fac[i] = fac[i - 1] * sm[i] % mod;inv[N] = qpow(fac[N], mod - 2);for (int i = N - 1; i >= 0; --i)inv[i] = inv[i + 1] * sm[i + 1] % mod;
}signed main() {ios::sync_with_stdio(0);cin.tie(0);gtf();solve();int T;cin >> T;while (T--) {int n, m;cin >> n >> m;if (n > m) swap(n, m);int l = 1, r = 0, ans = 1;while (l <= n) {r = min(n / (n / l), m / (m / l));ans = ans * qpow(fac[r] * inv[l - 1] % mod, (n / l) * (m / l)) % mod;l = r + 1;}cout << ans << "\n";}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/871664.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Servlet 详解!

一、Servlet简介 Servlet是Sun公司开发动态Web的一门技术。Sun公司在这些API中提供了一个Servlet接口,如果你想开发一个Servlet程序只需要完成如下两个步骤: 1、编写一个Java类实现Servlet接口。 2、把开发好的Java类部署到Web服务器中。 我们把实现了Servlet接口的Java程序叫…

图论/连通性

点边连通度:耳分解: 强连通有向图/边双联通无向图 从一个点出发,每次加入从集合出发回到集合,中间点不在集合内的环,一定能生成该图。 边双 强连通 双极定向:link 割空间与环空间互为正交补。 切边等价:模板 qoj1351CF1648F 树分解:也就是找到一种划分方式,使得每种划…

比特c语言-分支与循环

# 分支与循环 if语句 目录if语句ifeg:输入一个整数,判断是否为奇数elseeg:输入一个整数,判断是否为奇数,如果是奇数打印是奇数,否则打印偶数嵌套ifeg:输入一个人的年龄关系操作符条件操作符eg:使用条件操作符表示代码逻辑eg:使用条件表达式找两个数中较大值逻辑操作符…

NOIP 冲刺之——数据结构

\(\texttt{0x00}\) 前言 本篇文章主要记录笔者 NOIP 冲刺阶段复习的各种数据结构题型及 tricks ans tips,同时也用于及时复习与巩固。 那么,开始吧。 \(\texttt{0x01}\) 树状数组、线段树 知识点 \(1\):二维偏序 众所周知,逆序对可以用归并排序离线求,但是要求在线呢? 这…

windows 将docker desktop上镜像打包并通过资源管理器找到使用

在 Windows 上使用 Docker Desktop 时,可以通过以下步骤将 Docker 镜像保存为 .tar 文件,并通过资源管理器找到该文件:步骤 1:打开 Docker Desktop 确保 Docker Desktop 正在运行。如果未运行,请启动它。步骤 2:打开 PowerShell 或命令提示符按 Win + S,搜索 PowerShell…

解决方案 | office365/office 修复方法

打开控制面板,选择卸载程序,右键office 一般情况下,选择快速修复可以实现修复。如不行,那么选择联机修复‘。还是不行那就重装office

【pywinauto 库】启动PC端应用程序 - 上篇

一、简介 经过上一篇的学习、介绍和了解,想必小伙伴或者童鞋们,已经见识到pywinauto的强大了,今天继续介绍pywinauto,上一篇已经可以打开计算器了,这里宏哥再提供其他方法进行打开计算器和非电脑自带程序。pywinauto 可以启动电脑自带的应用程序,也可以启动电脑安装的应用…

【反EDR 】概要

一、什么是 EDR EDR 是“端点检测和响应”的缩写。它是部署在每台机器上的代理,用于观察操作系统生成的事件以识别攻击。如果检测到某些东西,它将生成警报并将其发送到 SIEM 或 SOAR,由人工分析师进行查看。“响应”是指在识别威胁后执行的操作,例如隔离主机,这不是本文的…

USB接口颜色都代表什么含义

手机充电器人人都有!充电器线颜色都不同!你知道不同颜色的USB接口的各个颜色都代表什么含义吗?大部分人都是不知道的,这篇文章让您 一目了然!建议收藏备用!以备不时之需!

Windows资源管理器Icon图标注入

免责声明 本文发布的工具和脚本,仅用作测试和学习研究,禁止用于商业用途,不能保证其合法性,准确性,完整性和有效性,请根据情况自行判断。简介 使用图标将 DLL 注入资源管理器的非正统和隐蔽方式 IconJector 这是一个Windows资源管理器DLL注入技术,使用Windows上的更改图…

ElasticSearch Query DSL(查询领域特定语言)

目录常用 DSL 关键字查询上下文相关度评分:_score源数据:_source数据源过滤器query 和 filter 上下文相关性评分 (relevance scores)query 的上下文filter 的上下文关于 query 和 filter 上下文的例子全文查询 (full text query)intervals 查询请求示例intervals的顶级参数ma…

ESP32 学习笔记(九)舵机实验

概念 舵机是一种位置(角度)伺服的驱动器,适用于那些需要角度不断变化并可以保持的控制系统。舵机只是一种通俗的叫法,其本质是一个伺服电机。 舵机有很多规格,但所有的舵机都有外接三根线,分别用棕、红、橙三种颜色进行区分,由于舵机品牌不同,颜色也会有所差异,棕色为…