【UOJ#33】【UR#2】树上GCD 有根树点分治 + 容斥原理 + 分块
#33. 【UR #2】树上GCD
有一棵$n$个结点的有根树$T$。结点编号为$1…n$,其中根结点为$1$。
树上每条边的长度为$1$。我们用$d(x,y)$表示结点$x,y$在树上的距离,$LCA(x,y)$表示$x,y$的最近公共祖先(即树中最深的既是$v$的祖先也是$u$的祖先的结点)。
对于两个结点$u,v(u≠v)(u≠v)$,令$a=LCA(u,v)$,定义$f(u,v)=gcd(d(u,a),d(v,a))$。
其中$gcd(x,y)$表示$x,y$的最大公约数,特别地,$gcd(0,x)=gcd(x,0)=x (x≠0)$。
对于所有$i∈{1,2,⋯,n−1}$,求出有多少对$(u,v) (u<v)$,满足$f(u,v)=i$。
输入格式
输入第一行包含一个正整数$n$。保证$n≥2$。
第$2$到$n$行中,第$i$行有一个正整数$pi$,表示结点$i$在树上的父亲是$pi$。保证$pi<i$。
输出格式
输出共$n−1$行,其中第$i$行表示对于$i$的答案。
C/C++ 输入输出 long long 时请用 %lld
。C++ 可以直接使用 cin/cout 输入输出。
样例一
input
5
1
2
2
1
output
8
2
0
0
样例二
input
8
1
2
3
4
1
6
6
output
16
9
2
1
0
0
0
样例三
见样例数据下载。
限制与约定
测试点编号 | n的规模 | 备注 |
---|---|---|
1 | n≤2000 | $pi$在${1,2,⋯,i−1}$中均匀随机 |
2 | n≤100000 | |
3 | n≤200000 | |
4 | n≤200000 | 除根结点外的每个结点至多拥有一个孩子 |
5 | n≤200000 | $pi$在$max{i−10,1}$到$i−1$之间均匀随机 |
6 | n≤50000 | 无 |
7 | n≤100000 | |
8 | n≤150000 | |
9 | n≤200000 | |
10 | n≤200000 |
时间限制:1s
空间限制:256MB
下载
Solution
这题是真的神....官方题解:UOJ Round#2 题解
Code
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
using namespace std;
int read()
{
int x=0,f=1; char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
return x*f;
}
#define maxn 200010
int n,m,fa[maxn],root,tmp,block,siz;
struct EdgeNode{int next,to;}edge[maxn<<1];
int head[maxn],cnt;
void add(int u,int v) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v;}
void insert(int u,int v) {add(u,v); add(v,u);}
int size[maxn],maxx[maxn]; bool visit[maxn];
void Getroot(int now,int fa)
{
size[now]=1; maxx[now]=0;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=fa && !visit[edge[i].to])
{
Getroot(edge[i].to,now);
size[now]+=size[edge[i].to];
maxx[now]=max(maxx[now],size[edge[i].to]);
}
maxx[now]=max(maxx[now],siz-size[now]);
if (maxx[now]<maxx[root]) root=now;
}
int num[maxn],deep[maxn];
void DFSdeep(int now,int fa)
{
num[deep[now]]++;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=fa && !visit[edge[i].to])
{
deep[edge[i].to]=deep[now]+1;
DFSdeep(edge[i].to,now);
}
if (deep[now]>tmp) tmp=deep[now];
}
long long S[maxn],Ans[maxn],ans[maxn],tim[maxn],Num[maxn],Tim[maxn],f[510][510];
void GetAns(int now)
{
visit[now]=1;
int maxd=0,Maxd=0;
for (int i=head[now]; i; i=edge[i].next)
if (!visit[edge[i].to] && edge[i].to!=fa[now])
{
deep[edge[i].to]=1;
tmp=0; DFSdeep(edge[i].to,now);
if (tmp>maxd) maxd=tmp;
for (int j=1; j<=tmp; j++)
for (int k=j; k<=tmp; k+=j)
tim[j]+=num[k];
for (int j=1; j<=tmp; j++)
Num[j]+=num[j],Ans[j]+=num[j],S[j]+=tim[j]*Tim[j],
Tim[j]+=tim[j],tim[j]=0,num[j]=0;
}
Num[0]=1;
int zz=0,ss=now;
for (int i=fa[now]; !visit[i]&&i; ss=i,i=fa[i])
{
tmp=0; zz++;
for (int j=head[i]; j; j=edge[j].next)
if (edge[j].to!=fa[i] && !visit[edge[j].to] && edge[j].to!=ss)
deep[edge[j].to]=1,DFSdeep(edge[j].to,i);
if (tmp>Maxd) Maxd=tmp;
for (int j=1; j<=tmp; j++)
for (int k=j; k<=tmp; k+=j)
tim[j]+=num[k];
int tt=tmp<block?tmp:block;
for (int j=1; j<=tt; j++)
{
if (f[j][zz%j]==-1)
{
f[j][zz%j]=0;
for (int k=(j-zz%j)%j; k<=maxd; k+=j)
f[j][zz%j]+=Num[k];
}
S[j]+=f[j][zz%j]*tim[j];
}
for (int j=block+1; j<=tmp; j++)
for (int k=(j-zz%j)%j; k<=maxd; k+=j)
S[j]+=Num[k]*tim[j];
for (int j=1; j<=tmp; j++) tim[j]=0,num[j]=0;
Ans[zz]++;
}
int l=1,r=0;
long long tmpans=0;
for (int i=2; i<=zz+maxd; i++)
tmpans+=r+1<i?Num[++r]:0,tmpans-=l+zz<i?Num[l++]:0,Ans[i]+=tmpans;
int tt=Maxd<block?Maxd:block;
for (int i=1; i<=tt; i++)
for (int j=0; j<=i-1; j++)
f[i][j]=-1;
for (int i=0; i<=maxd; i++) Num[i]=0,Tim[i]=0;
for (int i=head[now]; i; i=edge[i].next)
if (!visit[edge[i].to])
{
root=0;
siz=size[edge[i].to];
Getroot(edge[i].to,now);
GetAns(root);
}
}
void Freopen() {freopen("TreeGCD.in","r",stdin);freopen("TreeGCD.out","w",stdout);}
void Fclose() {fclose(stdin);fclose(stdout);}
int main()
{
n=read(); block=(int)sqrt(n);
for (int i=2; i<=n; i++) fa[i]=read(),insert(fa[i],i);
maxx[root]=0x7fffffff; siz=n; memset(f,-1,sizeof(f));
Getroot(1,0);
GetAns(root);
for (int i=n-1; i; i--)
{
ans[i]=S[i];
for (int j=i+i; j<=n-1; j+=i)
ans[i]-=ans[j];
}
for (int i=1; i<=n-1; i++) printf("%lld\n",ans[i]+Ans[i]);
return 0;
}
【UOJ#33】【UR#2】树上GCD 有根树点分治 + 容斥原理 + 分块的更多相关文章
- UOJ33 [UR #2] 树上GCD 【点分治】【容斥原理】【分块】
题目分析: 树上点对问题首先想到点分治.假设我们进行了点分治并递归地解决了子问题.现在我们合并问题. 我们需要找到所有经过当前重心$ c $的子树路径.第一种情况是LCA为当前重心$ c $.考虑以$ ...
- [UOJ UR #2]树上GCD
来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 看完题目,一般人都能想到 容斥稳了 .这样我们只要统计有多少点对满足gcd是i的倍数. 考虑长链剖分,每次合并的时候,假设我已经求出轻 ...
- 【uoj33】 UR #2—树上GCD
http://uoj.ac/problem/33 (题目链接) 题意 给出一棵${n}$个节点的有根树,${f_{u,v}=gcd(dis(u,lca(u,v)),dis(v,lca(u,v)))}$ ...
- 【UOJ#33】【UR #2】树上GCD(长链剖分,分块)
[UOJ#33][UR #2]树上GCD(长链剖分,分块) 题面 UOJ 题解 首先不求恰好,改为求\(i\)的倍数的个数,最后容斥一下就可以解决了. 那么我们考虑枚举一个\(LCA\)位置,在其两棵 ...
- 【UR #2】树上GCD
这道题是有根树点分治+烧脑的容斥+神奇的分块 因为是规定1为根,还要求LCA,所以我们不能像在无根树上那样随便浪了,必须规定父亲,并作特殊讨论 因为gcd并不好求,所以我们用容斥转化一下,求x为gcd ...
- 【BZOJ-3784】树上的路径 点分治 + ST + 堆
3784: 树上的路径 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 462 Solved: 153[Submit][Status][Discuss ...
- 【Luogu2664】树上游戏(点分治)
[Luogu2664]树上游戏(点分治) 题面 洛谷 题解 很好的一道点分治题. 首先直接点分治,考虑过每个分治重心的链的贡献. 我们从分治重心开始找每种颜色,强制令一种颜色只在其到分治重心的链上第一 ...
- [UOJ]#33. 【UR #2】树上GCD
题目大意:给定一棵有根树,边长均为1,对于每一个i,求树上有多少个点对,他们到lca距离的gcd是i.(n<=200,000) 做法:先容斥,求出gcd是i的倍数的点对,考虑长链剖分后从小到大合 ...
- UOJ#33. 【UR #2】树上GCD 点分治 莫比乌斯反演
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ33.html 题解 首先我们把问题转化成处理一个数组 ans ,其中 ans[i] 表示 d(u,a) 和 ...
随机推荐
- memcache 的内存管理介绍和 php实现memcache一致性哈希分布式算法
1 网络IO模型 安装memcached需要先安装libevent Memcached是多线程,非阻塞IO复用的网络模型,分为监听主线程和worker子线程,监听线程监听网络连接,接受请求后,将连接描 ...
- Web APP开发技巧总结
http://www.admin10000.com/document/6304.html 1. 当网站添加到主屏幕后再点击进行启动时,可隐藏地址栏(从浏览器跳转或输入链接进入并没有此效果) <m ...
- Ninject 学习杂记
IOC容器的DI实现并不依赖于方法调用拦截,而是通过DI容器内部自己通过反射的方式生成需要的类型实例,并调用实例的成员.然后再把实例返回给容器外部环境使用. Ninject本身及其扩展库,还针对特定的 ...
- Windows 8.1 新增控件之 CommandBar
上一篇为大家介绍了AppBar 的相关内容,本篇继续介绍CommandBar 的使用方法.与AppBar 相比而言,CommandBar 在开发使用方面较为单一,在按键布局上分为主控区(Primary ...
- SQL Server数据库代码指令简介
这些是比较常用的命令操作,事先声明,这些命令是不区分大小写的,我按照我的课本来总结用法和知识点,无用的章节自动省略. 没有一点数据库知识基础的可以等我录制视频,不然可能看不懂,视频链接:http:// ...
- Wabpack系列:在webpack+vue开发环境中使用echarts导致编译文件过大怎么办?
现象,在一个webpack+vue的开发环境中,npm install echarts --save了echarts,然后在vue文件中直接使用 import echarts from 'echart ...
- Reverse Words in a String
void reverseWords(string &s) { string res = "", tmp = ""; int l = s.length() ...
- MySql 中文乱码排查解决方案
MySQL会出现中文乱码的原因不外乎下列几点: server本身设定问题,例如还停留在latin1 table的语系设定问题(包含character与collation) 客户端程式(例如php)的连 ...
- Excel导入导出(篇二)
<body> <h3>一.Excel导入</h3> <h5>.模板下载:<a href="UpFiles/TemplateFiles/学 ...
- UltraEdit 编译输出中文乱码的解决办法
配置UE的时候,都告诉大家javac %n%e 接可以了,但是再运行的时候,会出现乱码,再加上-J-Duser.language=GBK就可以了,也就是下面的样子. javac -J-Duser. ...