【UOJ#33】【UR #2】树上GCD(长链剖分,分块)
【UOJ#33】【UR #2】树上GCD(长链剖分,分块)
题面
题解
首先不求恰好,改为求\(i\)的倍数的个数,最后容斥一下就可以解决了。
那么我们考虑枚举一个\(LCA\)位置,在其两棵不同的子树中选择两个点,那么贡献就是这两段的\(gcd\)。
那么发现要统计的东西类似于\(u\)的子树中,深度为\(d\)的点的个数,这个可以很容易的用长链剖分来维护,那么维护出这个数组之后就可以\(O(\log {dep})\)的对于贡献进行计算。然而这个复杂度是假的,因为你每次都需要一次\(O(\log dep)\),这个\(dep\)不能是长链的\(dep\),而这里却是长链长度,所以这样子复杂度不正确,
那么考虑别的方法,那就分块。
对于\(dis\ge \sqrt n\)的情况,因为其倍数的个数只有\(\sqrt n\)个,所以直接暴力统计,这部分全局的总复杂度是\(O(n\sqrt n)\)。
对于\(dis\lt \sqrt n\),维护\(g[i][j]\)表示当前重链的所有子树中,深度模\(i\)为\(j\)的点的个数,这个东西可以在因为\(dis\lt \sqrt n\),所以每次\(O(\sqrt n)\)的加入就行了,这样子全局复杂度也是\(O(n\sqrt n)\)。
这样子复杂度就做到了\(O(n\sqrt n)\)。
#include<iostream>
#include<cstdio>
using namespace std;
#define ll long long
#define MAX 200200
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Line{int v,next;}e[MAX];int h[MAX],cnt=1;
inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;}
int n,hson[MAX],md[MAX],dep[MAX];
void dfs1(int u,int ff)
{
md[u]=dep[u]=dep[ff]+1;
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;if(v==ff)continue;
dfs1(v,u);
if(md[v]>md[hson[u]])hson[u]=v;
}
if(hson[u])md[u]=md[hson[u]];
}
int tmp[MAX<<2],*f[MAX],*id=tmp,t[MAX],g[355][355],BLK=350;
ll ans[MAX],pre[MAX];
void dfs(int u,int ff,int fl)
{
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;if(v==ff||v==hson[u])continue;
f[v]=id;id+=md[v]-dep[v]+1;dfs(v,u,0);
}
if(hson[u])f[hson[u]]=f[u]+1,dfs(hson[u],u,1);
f[u][0]=1;int mx=md[u]-dep[u];
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;if(v==ff||v==hson[u])continue;
int s=md[v]-dep[v]+1;
for(int j=1;j<=s;++j)t[j]=f[v][j-1];
for(int j=1;j<=s;++j)
for(int k=j+j;k<=s;k+=j)
t[j]+=t[k];
for(int j=1;j<=s;++j)
if(j>BLK)for(int k=j;k<=mx;k+=j)ans[j]+=1ll*f[u][k]*t[j];
else ans[j]+=1ll*g[j][dep[u]%j]*t[j];
for(int j=0;j<s;++j)f[u][j+1]+=f[v][j];
for(int j=dep[v];j<=md[v];++j)
for(int k=1;k<=BLK;++k)
g[k][j%k]+=f[v][j-dep[v]];
}
for(int i=1;i<=BLK;++i)++g[i][dep[u]%i];
if(!fl)
for(int k=1;k<=BLK;++k)
for(int j=dep[u];j<=md[u];++j)
g[k][j%k]=0;
}
int main()
{
n=read();
for(int i=2,x;i<=n;++i)x=read(),Add(x,i);
dfs1(1,0);f[1]=id;id+=md[1]+1;dfs(1,0,0);
for(int i=n;i;--i)
for(int j=i+i;j<=n;j+=i)
ans[i]-=ans[j];
for(int i=2;i<=n;++i)pre[dep[i]-1]+=1;
for(int i=n;i;--i)pre[i]+=pre[i+1];
for(int i=1;i<n;++i)printf("%lld\n",ans[i]+pre[i]);
return 0;
}
【UOJ#33】【UR #2】树上GCD(长链剖分,分块)的更多相关文章
- CF1009F Dominant Indices(树上DSU/长链剖分)
题目大意: 就是给你一棵以1为根的树,询问每一个节点的子树内节点数最多的深度(相对于这个子树根而言)若有多解,输出最小的. 解题思路: 这道题用树链剖分,两种思路: 1.树上DSU 首先想一下最暴力的 ...
- UOJ#33-[UR #2]树上GCD【长链剖分,根号分治】
正题 题目链接:https://uoj.ac/problem/33 题目大意 给出\(n\)个点的一棵树 定义\(f(x,y)=gcd(\ dis(x,lca),dis(y,lca)\ )\). 对于 ...
- [UOJ UR #2]树上GCD
来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 看完题目,一般人都能想到 容斥稳了 .这样我们只要统计有多少点对满足gcd是i的倍数. 考虑长链剖分,每次合并的时候,假设我已经求出轻 ...
- hdu 4607 Park Visit(树上最长链)
求树上最长链:两遍搜索. 第一次从树上任意点开始,最远点必然是某一条最长链上的端点u. 第二次从u开始,最远点即该最长链的另一端点. 先在最长链上走,不足再去走支链. 把询问数m错打成n,狠狠wa了一 ...
- [HDU4607]Park Visit(树上最长链)
HDU#4607. Park Visit 题目描述 Claire and her little friend, ykwd, are travelling in Shevchenko's Park! T ...
- Codeforces 1009 F. Dominant Indices(长链剖分/树上启发式合并)
F. Dominant Indices 题意: 给一颗无向树,根为1.对于每个节点,求其子树中,哪个距离下的节点数量最多.数量相同时,取较小的那个距离. 题目: 这类题一般的做法是树上的启发式合并,复 ...
- LOJ3053 十二省联考2019 希望 容斥、树形DP、长链剖分
传送门 官方题解其实讲的挺清楚了,就是锅有点多-- 一些有启发性的部分分 L=N 一个经典(反正我是不会)的容斥:最后的答案=对于每个点能够以它作为集合点的方案数-对于每条边能够以其两个端点作为集合点 ...
- 【BZOJ3653】谈笑风生(长链剖分)
[BZOJ3653]谈笑风生(长链剖分) 题面 BZOJ 洛谷 权限题啊.... 题解 首先根据题目给的条件,发现\(a,b\)都要是\(c\)的父亲. 所以这三个点是树上的一条深度单增的链. 因为\ ...
- 2018牛客网暑假ACM多校训练赛(第七场)I Tree Subset Diameter 动态规划 长链剖分 线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round7-I.html 题目传送门 - https://www.n ...
随机推荐
- 英语AquilariaCrassna奇楠沉香
越南奇楠沉香Aquilaria crassna是瑞香科沉香属植物. 奇楠香被喻为沉香中的钻石,其与身俱来的香气,淡雅宜人,汇集天地阴阳五行之气,而成为唯一能通三界之香品.长久以来,它被视为一种珍贵罕有 ...
- 轻量级流程图控件GoJS示例连载(一):最小化
GoJS是一款功能强大,快速且轻量级的流程图控件,可帮助你在JavaScript 和 HTML5 Canvas程序中创建流程图,且极大地简化你的JavaScript / Canvas 程序. 慧都网小 ...
- 201871010107-公海瑜《面向对象程序设计(java)》第十五周学习总结
201871010107-公海瑜<面向对象程序设计(java)>第十五周学习总结 项目 内容 这个作业属于 ...
- 海量数据MySQL项目实战
主要内容包含 MySQL 典型数据库架构介绍.MySQL 主流数据库架构对比等理论性知识,然后从“订单.用户”两个项目实战,抛砖引玉,介绍亿级互联网业务数据库项目如何设计. MySQL 典型数据库架构 ...
- es6之后,真的不需要知道原型链了吗?
3月份几乎每天都能看到面试的人从我身边经过,前段时间同事聊面试话题提到了原型链,顿时激起了我在开始学习前端时很多心酸的回忆.第一次接触js的面向对象思想是在读<js高程设计>(红宝书)的时 ...
- nowcoder907B n的约数
题意 t次询问,每次给你一个数n,求在[1,n]内约数个数最多的数的约数个数 \(t \le 500,n \le 10^{19}\) 思路 首先可以想到将n质因数分解.即\(n= \prod\limi ...
- Git的回滚
Git常用命令: git add . #进行提交 git commit -m v2 #v2版本 cat index.html #查看文件内容已经变化 git log #获取到提交信息 git re ...
- 怎样删掉vc++ 对话框中的蓝色虚线框
选择"格式"--"切换辅助线"或者是快捷键alt+o , g
- [LeetCode] 884. Uncommon Words from Two Sentences 两个句子中不相同的单词
We are given two sentences A and B. (A sentence is a string of space separated words. Each word co ...
- [LeetCode] 70. Climbing Stairs 爬楼梯问题
You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...