懒得复制,戳我戳我

Solution:

  • 这题思路很神奇,首先你要知道这个毁灭树是怎么保证实现的:一句话就是如果该节点要被破坏,他的所有父节点就要被破坏,也就只要所有父节点的LCA被破坏就可以,所以我们就只用把这个点直接连向LCA。每个点都这么处理,最后面就是留下的一棵毁灭树,也就是\(i\)节点破坏后,它的子树都一定会被破坏
  • 上面这一点的详解戳我
  • 然后就是拓扑序的应用,我们读入并储存的是\(i\)节点连向的那些边,(没有连出的边的节点连向\(0\),不能连向自己,后面解释)然后拓扑排序,这是处理出的是原图节点中最后的节点在最前的拓扑序里(好像有点绕)
  • 我们就从图的最首位置开始处理(也就是拓扑序最后面的,这样可以保证后面的点可以处理父节点的LCA,并且不会影响LCA的结果,可以想想为什么)
  • 我们一边处理节点时,要处理出该节点在新的一棵树中的\(fa\)和\(depth\),这样就后面节点才能处理LCA
  • 然后我们就从叶子节点开始处理子树大小,我们只用把子树大小\(+1\)(这是加上自己),再加到父节点上。这时候我们就可以知道,树根的\(size\)会加到\(0\)去,不然会加到自己身上
  • 然后输出的时候就输出\(size-1\)就可以了,减去自己
  • 还有一个就是后面处理的时候都是用的拓扑序,下面就是自己犯的一个傻逼错误:

忘用team了,因为是处理新树上的值,拓扑序在前面的是更深的节点,这样才保证是从下至上处理size的####

for(int i=1;i<=n;i++){
size[ team[i] ]++;size[ fa[team[i]][0] ]+=size[ team[i] ];
}

Code:

//It is coded by Ning_Mew on 3.20
#include<bits/stdc++.h>
using namespace std; const int maxn=65534+10;
const int maxm=1e6+7; int n;
int team[maxn],ct=0;
int head[maxn],cnt=0;
struct EE{
int nxt,to;
}edge[maxm];
int fa[maxn][20],in[maxn],size[maxn],depth[maxn]; void add(int from,int to){
edge[++cnt].nxt=head[from];
edge[cnt].to=to;
head[from]=cnt;
}
void topsort(){
queue<int>q;
while(!q.empty())q.pop();
for(int i=1;i<=n;i++)if(in[i]==0)q.push(i),team[++ct]=i;
while(!q.empty()){
int u=q.front();q.pop();
for(int i=head[u];i!=0;i=edge[i].nxt){
int v=edge[i].to;
in[v]--;if(in[v]==0)q.push(v),team[++ct]=v;
}
}return;
}
int LCA(int x,int y){
if(depth[y]>depth[x])swap(x,y);
for(int i=16;i>=0;i--){
if(depth[ fa[x][i] ]>=depth[y])x=fa[x][i];
}if(y==x)return x;
for(int i=16;i>=0;i--){
if(fa[x][i]!=fa[y][i])x=fa[x][i],y=fa[y][i];
}return fa[x][0];
}
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
int box;scanf("%d",&box);
if(!box){add(i,0);continue;}
while(box){
add(i,box);in[box]++;
scanf("%d",&box);
}
}
topsort();
//for(int i=1;i<=n;i++)cout<<team[i]<<' ';cout<<endl;
for(int i=n;i>=1;i--){
int u,t;
u=team[i]; t=edge[ head[u] ].to;
for(int j=edge[ head[u] ].nxt;j!=0;j=edge[j].nxt){
t=LCA(t,edge[j].to);
}
//cout<<u<<' '<<t<<endl;
fa[u][0]=t;depth[u]=depth[t]+1;
for(int j=1;j<=16;j++)fa[u][j]=fa[ fa[u][j-1] ][j-1];
}
for(int i=1;i<=n;i++){
size[ team[i] ]++;size[ fa[team[i]][0] ]+=size[ team[i] ];
}
for(int i=1;i<=n;i++){printf("%d\n",size[i]-1);}
return 0;
}

【题解】 [ZJOI2012]灾难 (拓扑排序+LCA)的更多相关文章

  1. 【BZOJ2815】[ZJOI2012]灾难 拓扑排序+LCA

    [BZOJ2815][ZJOI2012]灾难 题目描述 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从 ...

  2. BZOJ2815:[ZJOI2012]灾难(拓扑排序,LCA)

    Description 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,如果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的生态灾难. 学过 ...

  3. 【bzoj2815】[ZJOI2012]灾难 拓扑排序+倍增LCA

    题目描述(转自洛谷) 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的生态灾难. 学过生物 ...

  4. 洛谷P2597 [ZJOI2012] 灾难 [拓扑排序,LCA]

    题目传送门 灾难 题目描述 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,如果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的生态灾难. ...

  5. 【bzoj2815】灾难[ZJOI2012](拓扑排序+lca)

    题目传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=2815 原版题解:http://fanhq666.blog.163.com/blog/st ...

  6. P2597 [ZJOI2012]灾难 拓扑排序

    这个题有点意思,正常写法肯定会T,然后需要优化.先用拓扑排序重构一遍树,然后进行一个非常神奇的操作:把每个点放在他的食物的lca上,然后计算的时候直接dfs全加上就行了.为什么呢,因为假如你的食物的l ...

  7. [BZOJ2815][ZJOI2012]灾难(拓扑排序/支配树)

    支配树目前只见到这一个应用,那就不独分一类,直接作为拓扑排序题好了. 每个点向所有食物连边,定义fa[x]为x的支配点,即离x最近的点,满足若fa[x]灭绝,则x也要灭绝. 这样,将fa[x]向x连边 ...

  8. BZOJ 2815: [ZJOI2012]灾难 拓扑排序+倍增LCA

    这种问题的转化方式挺巧妙的. Code: #include <bits/stdc++.h> #define N 100000 #define M 1000000 #define setIO ...

  9. [BZOJ2815][ZJOI2012]灾难 灭绝树+拓扑排序+lca

    灾难 [问题描述] 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,如果蚂蚱被他们捉灭绝了,那 么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的 生态灾难. 学过 ...

随机推荐

  1. c# SSH ,SFTP

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  2. Exp4 恶意代码分析 20155223

    Exp4 恶意代码分析 20155223 使用原生命令schtasks监视系统运行 在系统盘目录下建立脚本文件netstatlog.bat,包含以下命令: date /t >> c:\ne ...

  3. Luogu P1983 车站分级

    (一周没写过随笔了) 这道题有坑! 看到题目,发现这么明显(??)的要求顺序,还有什么想法,拓扑! 将每条路范围内等级大于等于它的点(不能重复(坑点1))和它连一条边,注意起点终点都要有(坑点2),然 ...

  4. mfc CTabCtrl

    知识点: CTabCtrl常用属性 CTabCtrl类常用成员函数 CTabCtrl代码示例 一.CTabCtrl控件属性 Bottom:底部样式 Vertical:垂直样式 与Bottom结合使用, ...

  5. 几个不常用的 Web API

    1. 设备震动 vibrate Navigator.vibrate() 方法使设备(有震动硬件)产生有频率的震动.若设备不支持震动,该方法将无效.若某震动方式已经在进行中(当该方法调用时),则前一个震 ...

  6. redis安装命令

    要进入redis的安装目录: cd 目录 安装命令:redis-server.exe --service-install redis.windows.conf --loglevel verbose 卸 ...

  7. Asp.Net_Mvc3.5语法_<%%>的用法

    一. <%%>这种格式实际上就是和asp的用法一样的,只是asp中里面是vbscript或 者javascript代码,而在asp.net中用的是.net平台下支持的语言.特别 注意:服务 ...

  8. 记录:Ubuntu 18.04 安装 tensorflow-gpu 版本

    狠下心来重新装了系统,探索一下 gpu 版本的安装.比较令人可喜的是,跟着前辈们的经验,还是让我给安装成功了.由于我是新装的系统,就像婴儿般纯净,所以进入系统的第一步就是安装 cuda,只要这个不出错 ...

  9. ats 安全

    Controlling Access ats可以配置为仅允许某些客户端使用代理缓存. 1. 为ip_allow.config添加一行,以获取允许访问ats的每个IP地址或IP地址范围; 2. traf ...

  10. ace -- about

    Built for Code Ace is an embeddable code editor written in JavaScript. It matches the features and p ...