【UOJ#33】【UR #2】树上GCD(长链剖分,分块)

题面

UOJ

题解

首先不求恰好,改为求\(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(长链剖分,分块)的更多相关文章

  1. CF1009F Dominant Indices(树上DSU/长链剖分)

    题目大意: 就是给你一棵以1为根的树,询问每一个节点的子树内节点数最多的深度(相对于这个子树根而言)若有多解,输出最小的. 解题思路: 这道题用树链剖分,两种思路: 1.树上DSU 首先想一下最暴力的 ...

  2. UOJ#33-[UR #2]树上GCD【长链剖分,根号分治】

    正题 题目链接:https://uoj.ac/problem/33 题目大意 给出\(n\)个点的一棵树 定义\(f(x,y)=gcd(\ dis(x,lca),dis(y,lca)\ )\). 对于 ...

  3. [UOJ UR #2]树上GCD

    来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 看完题目,一般人都能想到 容斥稳了 .这样我们只要统计有多少点对满足gcd是i的倍数. 考虑长链剖分,每次合并的时候,假设我已经求出轻 ...

  4. hdu 4607 Park Visit(树上最长链)

    求树上最长链:两遍搜索. 第一次从树上任意点开始,最远点必然是某一条最长链上的端点u. 第二次从u开始,最远点即该最长链的另一端点. 先在最长链上走,不足再去走支链. 把询问数m错打成n,狠狠wa了一 ...

  5. [HDU4607]Park Visit(树上最长链)

    HDU#4607. Park Visit 题目描述 Claire and her little friend, ykwd, are travelling in Shevchenko's Park! T ...

  6. Codeforces 1009 F. Dominant Indices(长链剖分/树上启发式合并)

    F. Dominant Indices 题意: 给一颗无向树,根为1.对于每个节点,求其子树中,哪个距离下的节点数量最多.数量相同时,取较小的那个距离. 题目: 这类题一般的做法是树上的启发式合并,复 ...

  7. LOJ3053 十二省联考2019 希望 容斥、树形DP、长链剖分

    传送门 官方题解其实讲的挺清楚了,就是锅有点多-- 一些有启发性的部分分 L=N 一个经典(反正我是不会)的容斥:最后的答案=对于每个点能够以它作为集合点的方案数-对于每条边能够以其两个端点作为集合点 ...

  8. 【BZOJ3653】谈笑风生(长链剖分)

    [BZOJ3653]谈笑风生(长链剖分) 题面 BZOJ 洛谷 权限题啊.... 题解 首先根据题目给的条件,发现\(a,b\)都要是\(c\)的父亲. 所以这三个点是树上的一条深度单增的链. 因为\ ...

  9. 2018牛客网暑假ACM多校训练赛(第七场)I Tree Subset Diameter 动态规划 长链剖分 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/NowCoder-2018-Summer-Round7-I.html 题目传送门 -  https://www.n ...

随机推荐

  1. 斗鱼刷弹幕js代码

    对于一个网络喷子(like me)来说,喷人必须高效. var script=document.createElement("script"); script.type=" ...

  2. JPA笔记2 OneToMany

    package one_to_many; import java.util.HashSet; import java.util.Set; import javax.persistence.Cascad ...

  3. office 所有后缀对应的 content-type

    后缀 MIME Type.doc application/msword.dot application/msword.docx application/vnd.openxmlformats-offic ...

  4. rt-thread can

    rt-thread stm32f10x-HAL can的驱动和应用.源码暂时还不支持,自己通过F4修改了一版 源码地址:https://gitee.com/gitee.thomas/rt_can rt ...

  5. composer install 出现“Please provide a valid cache path”

    本文背景:通过deployer部署PHP项目[deployer部署工具:https://deployer.org/] 问题:Php 的laravel框架中执行 composer install 后, ...

  6. kubernetes CRI 前世今生

    在学习kubernetes的过程中,我们会遇到CRI.CNI.CSI.OCI 等术语,本文试图先通过分析k8s目前默认的一种容器运行时架构,来帮助我们更好理解k8s 运行时背后设计逻辑.进而引出CRI ...

  7. C语言之整除

    除法运算符:/ 当除数和被除数都整形时,就是整除. 当浮点数和整数放到一起运算时,C语言会将整数转换成浮点数,然后进行浮点数的运算. #include<stdio.h> int main( ...

  8. 1.python进行if条件相等时候的条件

    在我们进行 if == 判断的时候!其中判断的条件: 1:其值是不是一样 3:其类型是否是一样 ###二者少了任何一个都不可以 >>> pwd = 23>>> cc ...

  9. 201871020225-牟星源《面向对象程序设计(java)》第十五周学习总结

    201871020225-牟星源<面向对象程序设计(java)>第十五周学习总结 项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ ...

  10. 【JSP】${pageContext.request.contextPath}

    取出部署的应用程序名或者是当前的项目名称 http://localhost:8080/demo1/a.jsp  ${pageContext.request.contextPath}或<%=req ...