首页 > 其他分享 >莫比乌斯函数学习笔记

莫比乌斯函数学习笔记

时间:2025-01-19 11:55:01浏览次数:1  
标签:lfloor frac int 乌斯 sum rfloor mu 笔记 莫比

\(\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;
}

标签:lfloor,frac,int,乌斯,sum,rfloor,mu,笔记,莫比
From: https://www.cnblogs.com/Rock-N-Roll/p/18679459

相关文章

  • ESP32 学习笔记(九)舵机实验
    概念舵机是一种位置(角度)伺服的驱动器,适用于那些需要角度不断变化并可以保持的控制系统。舵机只是一种通俗的叫法,其本质是一个伺服电机。舵机有很多规格,但所有的舵机都有外接三根线,分别用棕、红、橙三种颜色进行区分,由于舵机品牌不同,颜色也会有所差异,棕色为接地线,红色为电源正极......
  • AGC018做题笔记
    AtcoderGrandContest018B-SportsFestival题目大意有\(N\)个人参加\(M\)个项目的运动会,这\(M\)可以至多被删去\(M-1\)个。第\(i\)个人会参加序列\(a_i\)中第一个可以被参加的项目\(a_{i,j}\)。现在要使参加人数最多的项目人数最少,求这个最小人数。解题思......
  • Elasticsearch 笔记
    目录ES相关概念概述核心概念1)索引index2)类型type3)字段Filed4)映射mapping5)文档document6)集群cluster7)节点node8)分片和复制shards&replicas9)接近实时NRTElasticSearch字段的类型字段类型映射参数stringindex分析store存储Numericindex分析store存储dateindex分......
  • 折腾笔记[6]-使用rust绘制三维画面
    摘要使用rust绘制界面;界面包含一个三维坐标轴,使用鼠标旋转坐标轴,左侧显示对应的旋转向量,四元数等信息.UseRusttodrawtheinterface;theinterfacecontainsathree-dimensionalcoordinateaxis,whichcanberotatedusingthemouse,andthecorrespondingrotati......
  • 运算放大器应用电路设计笔记(四)
    动态范围表示正常工作时最小振幅与最大振幅的范围。例如,最小振幅为-14v,最大振幅为+14v,则动态范围为±14v,也有用绝对值或有效值表示振幅,最大电压与最小电压之比为动态范围,也称为多少dB。这时,最大振幅由电源电压决定,最小振幅由噪声或失调电压决定。确保动态范围的最简单方法......
  • 数学笔记
    定义一个形如\(x=a+bi\)的数为复数。上式中的\(i\)为虚数单位,有\(i\cdoti=-1\)。两个复数\(x\)和\(y\)相等,当且仅当\(x_a=y_a\)且\(x_b=y_b\)。对于一个复数\(x\),我们称\(a\)为\(x\)的实部,\(b\)为\(x\)的虚部。当\(a=0\)时,称\(x\)为纯虚数。......
  • NixOS使用笔记
    官方源:https://channels.nixos.org/清华源:https://mirrors.tuna.tsinghua.edu.cn/nix-channels本文使用清华源。升级系统官方文档:https://nixos.org/manual/nixos/stable/#sec-upgrading比如升级到24.11,首先升级到#sudonix-channel--addhttps://channels.nixos.org/nixo......
  • [数据结构学习笔记16] 线性查找(Linear Search)
    查找算法是指从一个集合里比如数组,列表,树里查找我们想要的值。我们从最简单的线性查找开始。线性查找,就是遍历集合里的元素,查看是否有和我们想要查找的值相同的,有则查找成功,没有则查找失败。比如:5,8,6,9,1,7,3,2,4我们要找3,那从5开始依次往后,到了第7个(下标6),我们找到了3。如果我们要找......
  • 计数问题学习笔记
    基础差得死,整版讲课课件能看懂的就\(10\%\),所以过来补一补。数学那一块差不多,计划单开一个博客。分类整理以下吧。卡特兰数问题引入有一个大小为\(n\timesn\)的网格图,每次从\((x,y)\)只能走一步到\((x+1,y)\)或\((x,y+1)\),求不走到对角线即\(y=x\)下方,但可以触碰对......
  • base中TCP/IP基础学习笔记
    base中的网络模型的学习笔记一.关于TCP/IP网络模型引言对于同一台设备上的进程间通信,有很多种方式,有管道、消息队列、共享内存、信号等方式,对于不同设备上的进程间通信,就需要网络通信,设备是多样的,所以要兼容各种各样的设备,就协商出了一套通用的网络协议。网络协议是分层......