【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组,需要保证在同一组中不存在一个点 ...
随机推荐
- http 结构初始化
简要而说:accept 到连接后 根据fd 构建一个connection 由于是 http : 重新封装为http-connection:同时设置fd的读回调: 回调函数根据是否是https/htt ...
- maven 笔记2
maven 中央工厂的位置:D:\dubbo\apache-maven-3.2.5\lib D:\dubbo\apache-maven-3.2.5\lib pom-4.0.0.xml reposito ...
- innodb之线程及IO相关参数介绍
引用链接:http://www.cnblogs.com/henglxm/p/4284504.html 1.IO THREAD: 负责IO的相关线程IO THREAD 1. 参数innodb_wri ...
- rhel8/centos8网络网卡设置ping不通,连接不上,各种问题
[解决问题]: 1-ping不通宿主机 2-ping不通外网 3-ping不通网关 4-网络中心VMnet8 VMnet1 VMnet0 不见了 5-rhel8网络设置全攻略 环境:win10宿主机+ ...
- 定位一个网络问题引起的ceph异常
前言 有一个ceph环境出现了异常,状态就是恢复异常的慢,但是所有数据又都在走,只是非常的慢,本篇将记录探测出问题的过程,以便以后处理类似的问题有个思路 处理过程 问题的现象是恢复的很慢,但是除此以外 ...
- 使用 ffmpeg 命令直播推流
CMD使用到的命令 D: -> 表示进入D盘cd /live -> 表示进入某个路径 执行条件 1.ffmpeg.exe 与视频放在同一目录下,纯英文目录2.执行以上命令进入到ffmpeg ...
- tp5 删除图片以及文件
控制器调用 /** * [delimg 删除单张图片] * @return [type] [description] */ public function delimg(){ if (request( ...
- 一次看完28个关于ES的性能调优技巧,很赞,值得收藏!
因为总是看到很多同学在说Elasticsearch性能不够好.集群不够稳定,询问关于Elasticsearch的调优,但是每次都是一个个点的单独讲,很多时候都是case by case的解答,本文简单 ...
- FL Studio通道窗口和步进音序器知识讲解
FL Studio中通道窗口是以样本为概念的音乐制作基础.通道窗口包含了步进音序器并结合了各种丰富的功能,可以通过通道窗口打开各通道的设置窗口.钢琴卷轴等.步进音序器在节奏制作方便表现出很好的优势,它 ...
- 缓存模式(Cache Aside、Read Through、Write Through、Write Behind)
目录 概览 Cache-Aside 读操作 更新操作 缓存失效 缓存更新 Read-Through Write-Through Write-Behind 总结 参考 概览 缓存是一个有着更快的查询速度 ...