约定:一棵树的深度定义为其中到根最远的点到根的距离

考虑节点$x$的答案:

任取一条直径,根据直径的性质,到$x$较远的直径端点一定是到$x$最远的点之一

由此,不难证明对于$x$独特的点,一定在$x$到"到$x$较远的直径端点"的路径上

分别以直径的两个端点为根(做两次),每一次求出$x$到根路径上所有对$x$独特的点的特产数,根据前面所述,两次的最大值即为最终答案

关于这件事情,维护一个"目前独特"的集合$S$,具体定义如下:

$k$到根路径上的节点$x$中(不包括$k$),满足不存在$k$子树外的节点$y\ne x$,使得$k$到$x$和$y$的距离相等("$k$子树"指"以$k$为根的子树")

当递归到节点$k$时,令$mx$为$k$子树深度,将$S$中到$k$距离不超过$mx$的节点都删除,此时$S$即$k$到根路径上所有与对$k$独特的点(所构成的集合),求出其中特产数即可

接下来,考虑递归儿子$son$,令$mx$为以$k$的其余儿子子树深度的最大值+1,将$S$中到$k$距离不超过$mx$的节点都删除,然后在加入$k$(显然满足条件),并在递归完$son$后还原$S$

显然,这样做的复杂度过高,我们需要优化实现:

将$S$用一个栈维护(栈顶深度最大),"将$S$中到$k$距离不超过$mx$的节点都删除"即不断弹出栈顶

将树长链剖分,(对于节点$k$)令$mx$为$k$子树深度,$cmx$为轻儿子子树深度的最大值+1,接下来交换之前操作的顺序,即先递归重儿子,再递归轻儿子,最后统计答案

考虑递归重儿子,即将$S$中到$k$距离不超过$cmx$的节点都删除,而递归轻儿子和统计答案都是将$S$中到$k$距离不超过$mx$的节点都删除,也就不需要还原,直接操作即可

统计答案后,也不需要还原$S$,假设以此法删除的点$x$,由于删去的是到$k$距离不超过$mx$的节点,即$k$子树内必然存在节点$y$使得$k$到$x$和$y$的距离相等

那么对$x$子树内、$k$子树外的点$z$,对于$z$来说$x$一定不"独特",只需要令$y'$为将$y$向上爬$k$到$lca(k,z)$的距离步后的节点,此时$z$到$x$和$y'$的距离显然相等

由此,发现最多在$S$中加入$o(n)$个元素,那么也即至多删除$o(n)$次,复杂度为$o(n)$

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 200005
4 struct Edge{
5 int nex,to;
6 }edge[N<<1];
7 stack<int>st;
8 int E,n,m,x,y,a[N],head[N],dep[N],l[N],cl[N],mx[N],tot[N],ans[N];
9 void add(int x,int y){
10 edge[E].nex=head[x];
11 edge[E].to=y;
12 head[x]=E++;
13 }
14 void dfs(int k,int fa,int s){
15 dep[k]=s;
16 mx[k]=l[k]=cl[k]=0;
17 for(int i=head[k];i!=-1;i=edge[i].nex)
18 if (edge[i].to!=fa){
19 dfs(edge[i].to,k,s+1);
20 int x=l[edge[i].to]+1;
21 if (l[k]<x){
22 mx[k]=edge[i].to;
23 swap(l[k],x);
24 }
25 cl[k]=max(cl[k],x);
26 }
27 }
28 void add(int k){
29 st.push(k);
30 if (++tot[a[k]]==1)ans[0]++;
31 }
32 void del(){
33 if (--tot[a[st.top()]]==0)ans[0]--;
34 st.pop();
35 }
36 void calc(int k,int fa){
37 if (fa)add(fa);
38 while ((!st.empty())&&(dep[k]-dep[st.top()]<=cl[k]))del();
39 if (mx[k])calc(mx[k],k);
40 while ((!st.empty())&&(dep[k]-dep[st.top()]<=l[k]))del();
41 ans[k]=max(ans[k],ans[0]);
42 for(int i=head[k];i!=-1;i=edge[i].nex)
43 if ((edge[i].to!=fa)&&(edge[i].to!=mx[k]))calc(edge[i].to,k);
44 if ((!st.empty())&&(st.top()==fa))del();
45 }
46 int main(){
47 scanf("%d%d",&n,&m);
48 memset(head,-1,sizeof(head));
49 for(int i=1;i<n;i++){
50 scanf("%d%d",&x,&y);
51 add(x,y);
52 add(y,x);
53 }
54 for(int i=1;i<=n;i++)scanf("%d",&a[i]);
55 dfs(1,0,0);
56 x=1;
57 for(int i=2;i<=n;i++)
58 if (dep[x]<dep[i])x=i;
59 dfs(x,0,0);
60 calc(x,0);
61 x=1;
62 for(int i=2;i<=n;i++)
63 if (dep[x]<dep[i])x=i;
64 dfs(x,0,0);
65 calc(x,0);
66 for(int i=1;i<=n;i++)printf("%d\n",ans[i]);
67 }

[loj3014]独特的城市的更多相关文章

  1. [LOJ3014][JOI 2019 Final]独特的城市——树的直径+长链剖分

    题目链接: [JOI 2019 Final]独特的城市 对于每个点,它的答案最大就是与它距离最远的点的距离. 而如果与它距离为$x$的点有大于等于两个,那么与它距离小于等于$x$的点都不会被计入答案. ...

  2. 【LOJ】#3014. 「JOI 2019 Final」独特的城市(长链剖分)

    LOJ#3014. 「JOI 2019 Final」独特的城市(长链剖分) 显然我们画一条直径,容易发现被统计的只可能是直径某个距离较远的端点到这个点的路径上的值 用一个栈统计可以被统计的点,然后我们 ...

  3. 题解 [JOI 2019 Final] 独特的城市

    题面 解析 首先有一个结论, 对一个点\(x\)有贡献的城市 肯定在它到离它较远的直径的端点的链上. 假设离它较远的端点是\(S\), 如果有一个点\(u\)不在\(x\)到\(S\)的链上, 却对\ ...

  4. loj 3014「JOI 2019 Final」独特的城市

    loj 我本来是直接口胡了一个意思一样的做法的,但是因为觉得有点假+实现要用并查集(?)就卡了好一会儿... 对于一个点\(x\)来说,独特的点一定在它的最长链上,如果有独特的点不在最长链上,那么最长 ...

  5. JOI 2019 Final合集

    JOI 2019 Final 合集 #3010. 「JOI 2019 Final」勇者比太郎 其实如果读懂题了就是水题了 题目就是让你求满足条件的\(JOI​\),使得\(O​\)在\(J​\)同行的 ...

  6. 洛谷 P 5 3 0 4 [GXOI/GZOI2019]旅行者

    题目描述 J 国有 n 座城市,这些城市之间通过 m 条单向道路相连,已知每条道路的长度. 一次,居住在 J 国的 Rainbow 邀请 Vani 来作客.不过,作为一名资深的旅行者,Vani 只对 ...

  7. JOI2013-2019

    代码自己去LOJ看 JOI2013 彩灯 把序列划分成若干极长交替列,那么最优的方案一定是将一个极长交替列翻转使得连续的三个极长交替列合成一个.计算相邻三个极长交替列长度的最大值即可. 搭乘IOI火车 ...

  8. 【JZOJ6419】模拟旅行&【BZOJ5506】【luoguP5304】旅行者

    description 某国有n座城市,这些城市之间通过m条单向道路相连,已知每条道路的长度. 不过,小X只对其中k座城市感兴趣. 为了更好地规划模拟旅行路线,提升模拟旅行的体验,小X想要知道他感兴趣 ...

  9. 「JOI2019 Final」解题报告

    传送门 「JOI2019 Final」勇者比太郎 看懂题就很简单了,后缀和随便维护一下就好了,别用树状数组强加一个\(\log\)就行. 「JOI2019 Final」画展 显然可以先把所有的画框按大 ...

随机推荐

  1. SpringBoot入门03-转发到Thymeleaf

    前言 Spring Boot不提倡使用jsp和用View层,而是使用Thymeleaf代替jsp,因为性能可以得到提升. 使用Thymeleaf要加入依赖 Thymeleaf不能直接被访问,它严格遵守 ...

  2. Apache Beam入门及Java SDK开发初体验

    1 什么是Apache Beam Apache Beam是一个开源的统一的大数据编程模型,它本身并不提供执行引擎,而是支持各种平台如GCP Dataflow.Spark.Flink等.通过Apache ...

  3. python T1119紧急措施

    2021-10-18 题目: 近日,一些热门网站遭受黑客入侵,这些网站的账号.密码及 email 的数据惨遭泄露.你在这些网站上注册若干账号(使用的用户名不一定相同),但是注册时使用了相同的 emai ...

  4. Postman实现SHA256withRSA签名

    @ 目录 获取pmlib 引入依赖bundle.js,有以下两种方式: 使用Pre-request Script对请求进行加签(具体加签字段请看自己项目) 获取pmlib 引入依赖bundle.js, ...

  5. Scrum Meeting 0605

    零.说明 日期:2021-6-5 任务:简要汇报两日内已完成任务,计划后两日完成任务 一.进度情况 组员 负责 两日内已完成的任务 后两日计划完成的任务 困难 qsy PM&前端 暂无 重新设 ...

  6. UltraSoft - Beta - Scrum Meeting 5

    Date: May 21st, 2020. Scrum 情况汇报 进度情况 组员 负责 今日进度 q2l PM.后端 修复了课程通知链接的bug Liuzh 前端 暂无 Kkkk 前端 增加消息中心板 ...

  7. BUAA SE 软件案例分析-CSDN

    Q A 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 个人博客作业-软件案例分析 我在这个课程的目标是 系统地学习软件工程开发知识,掌握相关流程和技术,提升 ...

  8. mybatis自定义分页拦截器

    最近看了一下项目中代码,发现系统中使用的mybatis分页使用的是mybatis自带的分页,即使用RowBounds来进行分页,而这种分页是基于内存分页,即一次查出所有的数据,然后再返回分页需要的数据 ...

  9. gson中TypeAdapter实现自定义序列化操作

    最近在项目中遇到这么一个问题,我们后台需要向前端返回一个 json 数据,就是将一个地理位置对象以json的格式返回到前台,但是这个地理位置对象中的经纬度是Double数据类型,项目中规定,如果经纬度 ...

  10. CCD摄像头视场角计算公式

    视场角大小和CCD传感器尺寸和镜头焦距有关: 水平视场角 = 2 × arctan(w / 2f); 垂直视场角 = 2 × arctan(h / 2f); 视场角 = 2 × arctan(d / ...