【Codeforces 809E】Surprise me!(莫比乌斯反演 & 虚树)
Description
给定一颗 \(n\) 个顶点的树,顶点 \(i\) 的权值为 \(a_i\)。求:
\]
其中 \(a\) 为一个 \(1\sim n\) 的排列。
Hint
\(1\le n\le 2\times 10^5\)
Solution
据说是套路题 然而我不会这个套路于是我觉得是神题 开一个 blog 记一下。
首先这个 \(\varphi(a_i\times a_j)\) 很烦,因为这样就不能分开计算了,于是考虑拆开:
& \varphi(x) = x\prod_{i=1}^s\left(1-\tfrac{1}{p_i} \right),\qquad \varphi(y) = y\prod_{i=1}^t\left(1-\tfrac{1}{q_i} \right), \qquad\varphi(xy) = xy\prod_{i=1}^k\left(1-\tfrac{1}{r_i}\right)\\
& \varphi(x)\times\varphi(y) = xy\prod_{i=1}^s\left(1-\tfrac{1}{p_i} \right)\prod_{i=1}^t\left(1-\tfrac{1}{q_i} \right)
\end{aligned}
\]
观察到 \(\varphi(x)\times \varphi(y)\) 相对 \(\varphi(xy)\) 多乘了一些东西,不难发现,其实就是 \(p\) 和 \(q\) 公共的部分:\(\gcd(x, y)\)。
那么我们除掉这些东西,有:
& \varphi(xy) = \frac{\varphi(x)\times \varphi(y)\times \gcd(x, y)}{\varphi(\gcd(x, y))} \\
& \sum_i \sum_j \varphi(a_i\times a_j)\times \text{dist}(i, j) = \sum_i \sum_j \frac{\varphi(a_i)\varphi(a_j) \gcd(a_i, a_j)}{\varphi(\gcd(a_i, a_j))}\text{dist}(i, j)
\end{aligned}
\]
\(\varphi(a_i\times a_j)\) 拆完了,但是冒出了个 \(\gcd\),怎么处理?
于是考虑枚举 \(\gcd\),记为 \(d\)。那么上式 \(=\)
& \sum_{d\le n} \sum_i\sum_j [d = \gcd(a_i, a_j)]\frac{\varphi(a_i)\varphi(a_j) d}{\varphi(d)}\text{dist}(i, j)\\
= & \sum_{d\le n} \frac{d}{\varphi(d)} \sum_i\sum_j [d = \gcd(a_i, a_j)]\varphi(a_i)\varphi(a_j)\text{dist}(i, j)
\end{aligned}
\]
然而 \([d = \gcd(a_i, a_j)]\) 还是不好搞。不过我们发现,如果这里是 \([d | \gcd(a_i, a_j)]\) 的话这个艾弗森括号就可以这样消:由于对于每个 \(d\) 都仅有满足 \(\gcd\) 为 \(d\) 的倍数时才有贡献,那么只要计算所有满足 \(d|a_i\) 的 \(i\) 点即可。
不难想到以建虚树的形式“抽出”这些点,由于 \(a\) 为 \(1\sim n\) 的一个排列,那么对于每个 \(d\),都有 \(\frac{n}{d}\) 个点满足条件。显然虚树的总点数不会超过 \(O(n\log n)\)。
扯了这么多,还是先得想办法把等号化成整除才行。
我们定义上面那个式子的后半部分为一个函数 \(f\),同时定义一个长得很像的 \(g\):
f(d) &= \sum_i\sum_j [d = \gcd(a_i, a_j)]\varphi(a_i)\varphi(a_j)\text{dist}(i, j)\\
g(d) &= \sum_i\sum_j [d | \gcd(a_i, a_j)]\varphi(a_i)\varphi(a_j)\text{dist}(i, j)
\end{aligned}
\]
可以观察到 \(g(x) = \sum_{x|d} f(d)\),那么可以通过莫比乌斯反演互转化:\(f(x) = \sum_{x|d} g(d)\mu\left(\frac d x\right)\)。只要算出了 \(g\),\(f\) 就能在 \(O(n\log n)\) 计算出来。
考虑 \(g\) 的计算。我们建好虚树 \(T\) 后,就先把艾弗森括号搞掉,为了方便我们记 \(v_i = \varphi(a_i)\):\(g(d) = \sum\limits_{i\in T}\sum\limits_{j\in T}v_iv_j\text{dist}(i, j)\)。
将 \(\text{dist}\) 拆成深度之和减去两倍 LCA 的深度之和的形式,那么有(\(d_i\) 表示点 \(i\) 在原树中的深度):
\]
其中 \(X\) 在求出子树 \(\sum v\) 之后可以轻松得到,\(Y\) 的话只要一波树形 dp 即可。
然后这题就完了,总时间复杂度 \(O(n\log^2 n)\)。
Code
/*
* Author : _Wallace_
* Source : https://www.cnblogs.com/-Wallace-/
* Problem : Codeforces 809E Surprise me!
*/
#include <algorithm>
#include <cctype>
#include <cstdio>
#include <vector>
using namespace std;
const int mod = 1e9 + 7;
const int N = 2e5 + 5;
typedef long long LL;
inline int read() {
int x(0); char c; while (!isdigit(c = getchar()));
do x = (x << 1) + (x << 3) + c - 48; while (isdigit(c = getchar()));
return x;
}
LL fastpow(LL a, LL b) {
if (!b) return 1ll;
LL t = fastpow(a, b / 2);
if (b & 1) return t * t % mod * a % mod;
else return t * t % mod;
}
LL inv(LL x) {
return fastpow(x % mod, mod - 2);
}
int p[N], tot = 0;
bool flag[N];
int mu[N], phi[N];
void sieve(int n) {
mu[1] = phi[1] = 1;
for (int i = 2; i <= n; i++) {
if (!flag[i]) mu[i] = -1, phi[i] = i - 1, p[++tot] = i;
for (int j = 1; j <= tot && i * p[j] <= n; j++) {
flag[i * p[j]] = 1;
if (i % p[j] == 0) { mu[i * p[j]] = 0, phi[i * p[j]] = phi[i] * p[j]; break; }
mu[i * p[j]] = -mu[i], phi[i * p[j]] = phi[i] * (p[j] - 1);
}
}
}
int n, a[N], loc[N];
vector<int> adj[N];
LL f[N], g[N];
int fa[N], siz[N], dep[N];
int wson[N], wtop[N];
int dfn[N], timer = 0;
void dfs1(int x, int f) {
dep[x] = dep[fa[x] = f] + 1, siz[x] = 1;
for (auto y : adj[x]) if (y != f) {
dfs1(y, x), siz[x] += siz[y];
if (siz[wson[x]] < siz[y]) wson[x] = y;
}
}
void dfs2(int x, int t) {
wtop[x] = t, dfn[x] = ++timer;
if (wson[x]) dfs2(wson[x], t);
for (auto y : adj[x]) if (y != fa[x] && y != wson[x])
dfs2(y, y);
}
int lca(int x, int y) {
while (wtop[x] != wtop[y]) {
if (dep[wtop[x]] < dep[wtop[y]]) swap(x, y);
x = fa[wtop[x]];
}
return dep[x] < dep[y] ? x : y;
}
bool tag[N];
vector<int> vir[N];
template<bool d> inline void link(vector<int>* g, int x, int y) {
g[x].push_back(y); if (d) g[y].push_back(x);
}
LL sum[N], X, Y;
void dfs(int x) {
sum[x] = tag[x] ? a[x] : 0ll;
for (auto y : vir[x]) {
dfs(y);
(Y += sum[x] * sum[y] % mod * dep[x] % mod) %= mod;
(sum[x] += sum[y]) %= mod;
}
vir[x].clear(), tag[x] = 0;
}
LL solve(int rt, int* vtx, int cnt) {
X = Y = 0ll, dfs(rt);
for (int i = 1; i <= cnt; i++) (X += a[vtx[i]] * 1ll * dep[vtx[i]] % mod * sum[rt] % mod) %= mod;
(Y *= 2) %= mod;
for (int i = 1; i <= cnt; i++) (Y = Y + a[vtx[i]] * 1ll * a[vtx[i]] % mod * dep[vtx[i]] % mod + mod) % mod;
(X *= 2) %= mod, (Y *= 2) %= mod;
return (X - Y + mod) % mod;
}
signed main() {
n = read(), sieve(n);
for (int i = 1; i <= n; i++) loc[a[i] = read()] = i, a[i] = phi[a[i]];
for (int i = 1, u, v; i < n; i++) u = read(), v = read(), link<1>(adj, u, v);
dfs1(1, 0), dfs2(1, 1);
for (int d = 1; d <= n; d++) {
static int vtx[N]; int cnt = 0;
for (int x = d; x <= n; x += d) tag[vtx[++cnt] = loc[x]] = 1;
sort(vtx + 1, vtx + 1 + cnt, [](int a, int b) { return dfn[a] < dfn[b]; });
static int stk[N]; int top = 0;
stk[top = 1] = vtx[1];
for (int i = 2; i <= cnt; i++) {
int x = vtx[i], l = lca(x, stk[top]);
for (; top > 1 && dep[l] <= dep[stk[top - 1]]; --top) link<0>(vir, stk[top - 1], stk[top]);
if (stk[top] != l) link<0>(vir, l, stk[top]), stk[top] = l;
stk[++top] = x;
}
for (; top > 1; --top) link<0>(vir, stk[top - 1], stk[top]);
g[d] = solve(stk[1], vtx, cnt);
}
for (int x = 1; x <= n; x++)
for (int d = x; d <= n; d += x)
(f[x] += g[d] * mu[d / x] + mod) %= mod;
LL ans = 0ll;
for (int i = 1; i <= n; i++) (ans += i * inv(phi[i]) % mod * f[i] % mod) %= mod;
printf("%lld\n", ans * inv(n * 1ll * (n - 1)) % mod);
return 0;
}
【Codeforces 809E】Surprise me!(莫比乌斯反演 & 虚树)的更多相关文章
- Codeforces.809E.Surprise me!(莫比乌斯反演 虚树)
题目链接 \(Description\) 给定一棵树,求\[\frac{1}{n(n-1)/2}\times\sum_{i\in[1,n],j\in[1,n],i\neq j}\varphi(a_i\ ...
- Codeforces 809E Surprise me! [莫比乌斯反演]
洛谷 Codeforces 非常套路的一道题,很适合我在陷入低谷时提升信心-- 思路 显然我们需要大力推式子. 设\(p_{a_i}=i\),则有 \[ \begin{align*} n(n-1)an ...
- YbtOJ#723-欧拉之树【莫比乌斯反演,虚树】
正题 题目链接:http://www.ybtoj.com.cn/contest/121/problem/2 题目大意 给出\(n\)个点的一棵树,每个点有一个权值\(a_i\),求 \[\sum_{i ...
- 【BZOJ3529】数表(莫比乌斯反演,树状数组)
[BZOJ3529]数表(莫比乌斯反演,树状数组) 题解 首先不管\(A\)的范围的限制 要求的东西是 \[\sum_{i=1}^n\sum_{j=1}^m\sigma(gcd(i,j))\] 其中\ ...
- 【bzoj3529】[Sdoi2014]数表 莫比乌斯反演+离线+树状数组
题目描述 有一张n×m的数表,其第i行第j列(1 <= i <= n ,1 <= j <= m)的数值为能同时整除i和j的所有自然数之和.给定a,计算数表中不大于a的数之和. ...
- Codeforces 809E - Surprise me!(虚树+莫比乌斯反演)
Codeforces 题目传送门 & 洛谷题目传送门 1A,就 nm 爽( 首先此题一个很棘手的地方在于贡献的计算式中涉及 \(\varphi(a_ia_j)\),而这东西与 \(i,j\) ...
- CF809E Surprise me! 莫比乌斯反演、虚树
传送门 简化题意:给出一棵\(n\)个点的树,编号为\(1\)到\(n\),第\(i\)个点的点权为\(a_i\),保证序列\(a_i\)是一个\(1\)到\(n\)的排列,求 \[ \frac{1} ...
- 【CF809E】Surprise me! 树形DP 虚树 数学
题目大意 给你一棵\(n\)个点的树,每个点有权值\(a_i\),\(a\)为一个排列,求 \[ \frac{1}{n(n-1)}\sum_{i=1}^n\sum_{j=1}^n \varphi(a_ ...
- CodeCraft-19 and Codeforces Round #537 (Div. 2) E 虚树 + 树形dp(新坑)
https://codeforces.com/contest/1111/problem/E 题意 一颗有n个点的树,有q个询问,每次从树挑出k个点,问将这k个点分成m组,需要保证在同一组中不存在一个点 ...
随机推荐
- Selective Acknowledgment 选项 浅析 1
抓包的时候,发现 tcp 三次握手中一般会有几个options 一个是mss 一个是ws 一个sack perm 这次主要是来说一说 sack 这个选项: 1. 只重传超时的数据包,比较实用与后 ...
- 主动关闭 time-wait 2msl 处理
先上传后面整理 /* * This routine is called by the ICMP module when it gets some * sort of error condition. ...
- MySQL索引结构之Hash索引、full-text全文索引(面)
Hash索引 主要就是通过Hash算法(常见的Hash算法有直接定址法.平方取中法.折叠法.除数取余法.随机数法),将数据库字段数据转换成定长的Hash值,与这条数据的行指针一并存入Hash表的对应位 ...
- binary hacks读数笔记(file命令与magic file)
file命令的作用是用于检验文件的类型,并打印至终端.file命令检验文件类型按以下顺序来完成: 检验文件系统(Filesystem)中支持的文件类型. 检验magic file规则. 检验文件内容的 ...
- NO.A.0002——Git简史及安装教程/创建本地仓库/提交项目到本地仓库/误删还原
一.Git简史及同类产品对比: 1.git简史: 同生活中的许多伟大事件一样,Git 诞生于一个极富纷争大举创新的年代.Linux 内核开源项目有着为数众广的参与者.绝大多数的 Linu ...
- FL studio系列教程(四):如何利用FL Studio进行音乐合并
FL Studio20是Fruity Loops Studio的简称,也叫做水果音乐制作软件.它是一款功能十分强大的音乐制作软件,将作曲.编曲.混音.录音.大碟等功能集合一体,外接MIDI即可成为一个 ...
- Lambda表达式(一)入门认识篇
Lambda表达式(一)入门认识篇 Lambda简介 Lambda 表达式是 JDK8 的一个新特性,可以取代大部分的匿名内部类,写出更优雅的 Java 代码,尤其在集合的遍历和其他集合操作中,可以极 ...
- 【P4211 LNOI2014】LCA——树链剖分 +询问离线
(7.16晚)更完先在B站颓一会儿-- --------------------------------------------------------------- (以下为luogu题面) 题目描 ...
- mysql 数据文件
mysql8.0取消了frm文件 . ibd数据和索引
- 论如何优雅的抛出SpringBoot注解的异常
平时我们在写代码的时候肯定要进行很多参数验证,最开始的时候我们一般都是这样处理的 如下图 看起来好像也没什么,但是 如果参数多了呢?你就会看到这样的校验 OMG!!! 有没有感觉稍微有点视觉 ...