首页 > 其他 > 详细

2101 可达性统计

时间:2018-08-21 22:07:22      阅读:194      评论:0      收藏:0      [点我收藏+]
【题目描述】
    给定一张N个点M条边的有向无环图,分别统计从每个点出发能够到达的点的数量。N,M≤30000。
【题目链接】

2101 可达性统计

【算法】

拓扑排序之后逆序计算,bitset状态压缩模拟集合的并操作。

【代码】
#include <bits/stdc++.h>
using namespace std;
int n,m,tot,cnt;
struct edge{ int to,next; }e[30010];
int head[30010],topn[30010],deg[30010];
bitset <30010> s[30010];
inline int read() {
    int x=0,f=1; char c=getchar();
    while(c<‘0‘||c>‘9‘) { if(c==‘-‘) f=-1; c=getchar(); }
    while(c>=‘0‘&&c<=‘9‘) { x=x*10+c-‘0‘; c=getchar(); }
    return x*f;
}
void add(int x,int y) {
    e[++tot].to=y,e[tot].next=head[x];
    head[x]=tot,deg[y]++;
}
void topsort() {
    queue<int> q;
    for(int i=1;i<=n;i++) if(!deg[i]) q.push(i);
    while(q.size()) {
        int x=q.front(); q.pop();
        topn[++cnt]=x;
        for(int i=head[x];i;i=e[i].next) {
            int to=e[i].to; deg[to]--;
            if(!deg[to]) q.push(to);
        }
    }
}
int main() {
    n=read(),m=read();
    for(int i=1;i<=m;i++) {
        int a,b;
        a=read(),b=read();
        add(a,b);
    }
    topsort();
    for(int i=cnt;i>=1;i--) {
        int x=topn[i];
        s[x][x]=1;
        for(int j=head[x];j;j=e[j].next) {
            int to=e[j].to;
            s[x]|=s[to];
        }
    }
    for(int i=1;i<=n;i++) printf("%d\n",s[i].count());
    return 0;
}

2101 可达性统计

原文:https://www.cnblogs.com/Willendless/p/9514511.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!