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

  题目传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=2815

  原版题解:http://fanhq666.blog.163.com/blog/static/8194342620124274154996/

  首先这个生态系统里可能有多个生产者,所以我们可以让生产者吃一种构造出来的食物——阳光。

  这道题的图没有环(不然还怎么叫食物网),所以我们可以用拓扑排序把从阳光到各级消费者的先后顺序求出来。

  我们可以建出一个结构(可以证明这个结构是一棵树),生物x会因为生物y的灭绝而灭绝,当且仅当y是x的祖先。

  因此,如果一种生物会灭绝,只能是因为它的所有食物的lca灭绝了,否则至少有一种食物不会灭绝,这种生物也能活下去,所以这种生物在树上的位置就是它所有食物的lca的孩子。

  然后我们把树建出来之后每个节点的答案就是以他为根的子树的节点个数-1。

  代码:

#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<ctime>
#include<algorithm>
#include<queue>
#include<vector>
#define ll long long
ll read()
{
    ll tmp=0; char f=1,c=getchar();
    while(c<‘0‘||‘9‘<c){if(c==‘-‘)f=-1; c=getchar();}
    while(‘0‘<=c&&c<=‘9‘){tmp=tmp*10+c-‘0‘; c=getchar();}
    return tmp*f;
}
using namespace std;
int fir[70010]={0},ne[1000010],to[1000010],x[1000010],y[1000010];
int q[70010],vis[70010]={0};
ll in[70010]={0},cnt[70010];
int fa[70010][20],dep[70010];
int n,m,tot=0;
void add(int x,int y){to[++tot]=y; ne[tot]=fir[x]; fir[x]=tot;}
int lca(int x,int y)
{
    int i;
    if(dep[x]<dep[y]){int tmp=x; x=y; y=tmp;}
    for(i=(int)(log(n)/log(2)+1);i>=0;i--)if(dep[x]-(1<<i)>=dep[y])x=fa[x][i];
    if(x==y)return x;
    for(i=(int)(log(n)/log(2)+1);i>=0;i--)if(fa[x][i]!=fa[y][i])x=fa[x][i],y=fa[y][i];
    return fa[x][0];
}
int main()
{
    int i,j;
    n=read();
    for(i=1;i<=n;i++){
        int k=read();
        if(!k)add(0,i),in[i]=1,x[tot]=i,y[tot]=0;
        while(k){++in[i]; add(k,i); x[tot]=i; y[tot]=k; k=read();}
    }
    int h=0,t=0; q[0]=0; vis[0]=1;
    while(h<=t){
        for(i=fir[q[h]];i;i=ne[i]){
            --in[to[i]];
            if(!in[to[i]])q[++t]=to[i],in[to[i]]=1<<30;
        }
        ++h;
    }
    m=tot; tot=0;
    for(i=1;i<=n;i++)fir[i]=0;
    for(i=1;i<=m;i++)add(x[i],y[i]);
    for(i=0;i<=n;i++)
        for(j=0;1<<j<=n;j++)
            fa[i][j]=-1;
    fa[0][0]=-1; dep[0]=0;
    for(i=1;i<=n;i++){
        int p=-1;
        for(j=fir[q[i]];j;j=ne[j])
            if(p==-1)p=to[j];else p=lca(p,to[j]);
        fa[q[i]][0]=p; dep[q[i]]=dep[p]+1;
        for(j=1;1<<j<=n&&fa[fa[q[i]][j-1]][j-1]>=0;j++)fa[q[i]][j]=fa[fa[q[i]][j-1]][j-1];
    }
    for(i=1;i<=n;i++)cnt[i]=1;
    for(i=n;i;i--)cnt[fa[q[i]][0]]+=cnt[q[i]];
    for(i=1;i<=n;i++)
        printf("%lld\n",cnt[i]-1);
}

bozj2815

时间: 2024-10-12 19:59:46

【bzoj2815】灾难[ZJOI2012](拓扑排序+lca)的相关文章

【BZOJ2815】[ZJOI2012]灾难 拓补排序+LCA

[BZOJ2815][ZJOI2012]灾难 题目描述 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的生态灾难. 学过生物的阿米巴告诉小强,草原是一个极其稳定的生态系统.如果蚂蚱灭绝了,小鸟照样可以吃别的虫子,所以一个物种的灭绝并不一定会引发重大的灾难. 我们现在从专业一点的角度来看这个问题.我们用一种叫做食物网的有向图来描述生物之间的关系: 一个食物网有N个点,代表N种生物,如果生物

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

题目描述(转自洛谷) 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,果蚂蚱被他们捉灭绝了,那么吃蚂蚱的小鸟就会饿死,而捕食小鸟的猛禽也会跟着灭绝,从而引发一系列的生态灾难. 学过生物的阿米巴告诉小强,草原是一个极其稳定的生态系统.如果蚂蚱灭绝了,小鸟照样可以吃别的虫子,所以一个物种的灭绝并不一定会引发重大的灾难. 我们现在从专业一点的角度来看这个问题.我们用一种叫做食物网的有向图来描述生物之间的关系: 一个食物网有N个点,代表N种生物,如果生物x可以吃生物y,那么从y向x连一个

P2597 [ZJOI2012]灾难 拓扑排序

这个题有点意思,正常写法肯定会T,然后需要优化.先用拓扑排序重构一遍树,然后进行一个非常神奇的操作:把每个点放在他的食物的lca上,然后计算的时候直接dfs全加上就行了.为什么呢,因为假如你的食物的lca死了,你就很自然的死了.这个题还要加一个超级源点,一开始我没加只拿了20分.但是不知道为什么,后来想到有可能一个兄弟能吃两个生产者,在查询lca的时候会跪...剩下就没啥了,顺便练一下倍增求lca. 题干: 题目描述 阿米巴是小强的好朋友. 阿米巴和小强在草原上捉蚂蚱.小强突然想,如果蚂蚱被他们

hihocoder 1343 : Stable Members【拓扑排序】

hihocoder #1343:题目 解释:一个学习小组,一共有N个学员,一个主管.每个学员都有自己的导师(一个或者多个),导师可以是其他学员也可以是主管.每周学员都要把自己的学习报告和收到的报告提交给自己的导师,这个团队设计很合理,没有回环(投递出去的报告不会回到自己手中),并且所有的报告最终都会投递到主管那里.但这个团队中有的学员会因为其他某个学员不工作而导致报告无法提交到主管手中,我们称这种学员为不可靠的.而不受某个学员不工作而影响到报告提交的就是可靠学员.问题就是找出可靠学员的数量. 输

【基础练习】【拓扑排序】codevs3294 车站分级题解

题目来源:NOIP2013 普及第四题 题目描写叙述 Description 一条单向的铁路线上,依次有编号为1, 2, -, n的n个火车站.每一个火车站都有一个级别,最低为1级.现有若干趟车次在这条线路上行驶.每一趟都满足例如以下要求:假设这趟车次停靠了火车站x.则始发站.终点站之间全部级别大于等于火车站x的都必须停靠. (注意:起始站和终点站自然也算作事先已知须要停靠的网站) 比如,下表是5趟车次的执行情况.当中.前4趟车次均满足要求,而第5趟车次因为停靠了3号火车站(2级)却未停靠途经的

拓扑排序讲解

在这里我们要说的拓扑排序是有前提的 我们在这里说的拓扑排序是基于有向无环图的!!!. (⊙o⊙)…我所说的有向无环图都知道是什么东西吧.. 如果不知道,我们下面先来来说说什么是有向无环图. 所谓有向无环图,顾名思义是不存在环的有向图(至于有向图是什么不知道的在前面我们有一个图论讲解上都有). 点的入度:以这个点为结束点的边数. 点的出度:以这个点为出发点的边的条数. 拓扑序就是对于一个节点的一个排列,使得(u,v)属于E,那么u一定出现在v的前面.然而拓扑排序就是一个用来求拓扑序的东西. 对于左

CSU 1804: 有向无环图(拓扑排序)

http://acm.csu.edu.cn/csuoj/problemset/problem?pid=1804 题意:…… 思路:对于某条路径,在遍历到某个点的时候,之前遍历过的点都可以到达它,因此在这个时候对答案的贡献就是∑(a1 + a2 + a3 + ... + ai) * bv,其中a是之前遍历到的点,v是当前遍历的点. 这样想之后就很简单了.类似于前缀和,每次遍历到一个v点,就把a[u]加给a[v],然后像平时的拓扑排序做就行了. 1 #include <bits/stdc++.h>

7-9-有向图无环拓扑排序-图-第7章-《数据结构》课本源码-严蔚敏吴伟民版

课本源码部分 第7章  图 - 有向无环图拓扑排序 ——<数据结构>-严蔚敏.吴伟民版        源码使用说明  链接??? <数据结构-C语言版>(严蔚敏,吴伟民版)课本源码+习题集解析使用说明        课本源码合辑  链接??? <数据结构>课本源码合辑        习题集全解析  链接??? <数据结构题集>习题解析合辑        本源码引入的文件  链接? Status.h.SequenceStack.c.ALGraph.c    

hihoCoder 1175:拓扑排序二

题目链接: http://hihocoder.com/problemset/problem/1175 题目难度:一星级(简单题) 今天闲来无事,决定刷一道水题.结果发现这道水题居然把我卡了将近一个钟头. 最后终于调通了.总结起来,原因只有一个:不够仔细. 思路不用细说了,就是拓扑排序的简单应用.然而,一些不起眼的细节才是让你掉坑里的真正原因. 猜猜哪儿可能出bug? // A simple problem, but you can't be too careful with it. #inclu