[tarjan缩点] 洛谷P2746 [USACO5.3]校园网Network of Schools

一开始完全没有搞懂题目的意思就下手,但是居然还AC了两个点?

仔细审视了一下题目的意思,发现题目并不难。

对于第一问,我们只需要求缩点后,入度为 0 的点的数量就可以了。

对于第二问,我们的目标是要求缩点后的所有点互相联通(因为只有这样,任选一个点才能互相到达)
我们转换一下含义:缩点后的所有点只有入度和出度都大于0 才为互相联通
所以第二问我们只需要对入度为0的点和出度0的点做个比较,谁大取谁的值输出。

坑点:需要特判一下只有一个联通块的时候,否则会出错。

#include <cstdio>
#include <algorithm>
#include <cstring>
#include <stack>
using std::stack;

const int N = 233;

struct node{
    int u,v,next;
    node(){}
    node(int _u,int _v,int _next){
        u = _u;
        v = _v;
        next = _next;
    }
}Edge[N*N];

int head[N*N],Count;
int dfn[N],low[N];
//scc_cnt是联通块的个数(缩点后的点的数量
//sccno是缩点后的序号
int scc_cnt,step,sccno[N];
int n;

void AddEdge(int u,int v){
    Count++;
    Edge[Count] = node(u,v,head[u]);
    head[u] = Count;
}

void init(){
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        int x;
        while(scanf("%d",&x) && x!=0){
            AddEdge(i,x);
        }
    }
}

stack<int>S;
void dfs(int x){
    dfn[x] = low[x] = ++step;
    S.push(x);
    for(int i = head[x];i;i=Edge[i].next){
        int v = Edge[i].v;
        if(!dfn[v]){
            dfs(v);
            low[x] = std::min(low[x],low[v]);
        }
        else if(!sccno[v]) low[x] =std::min(low[x],dfn[v]);
    }
    if(low[x] == dfn[x]){
        scc_cnt++;
        while(1){
            int u = S.top();
            S.pop();
            sccno[u] = scc_cnt;
            if(x==u) break;
        }
    }
}

int To[N],Out[N];

int main(){
    init();
    for(int i=1;i<=n;i++){
        if(!dfn[i]) dfs(i);
    }
    for(int i=1;i<=Count;i++){
        if(sccno[Edge[i].u]!=sccno[Edge[i].v]){
            Out[sccno[Edge[i].u]]++;
            To[sccno[Edge[i].v]]++;
        }
    }
    //To入度 Out出度
    int ask1=0,ask2=0;
    for(int i=1;i<=scc_cnt;i++){
        if( !To[i] ) ask1++;
        if( !Out[i] ) ask2++;
    }
    int End = std::max(ask1,ask2);
    if(scc_cnt==1){
        printf("1\n0\n");
    }
    else{
        printf("%d\n%d\n",ask1,End);
    }
    return 0;

}
时间: 2024-10-10 13:28:13

[tarjan缩点] 洛谷P2746 [USACO5.3]校园网Network of Schools的相关文章

P2746 [USACO5.3]校园网Network of Schools// POJ1236: Network of Schools

P2746 [USACO5.3]校园网Network of Schools// POJ1236: Network of Schools 题目描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软件(称作“接受学校”).注意即使 B 在 A 学校的分发列表中, A 也不一定在 B 学校的列表中. 你要写一个程序计算,根据协议,为了让网络中所有的学校都用上新软件,必须接受新软件副本的最少学校数目(子任务 A).更进一步,我们想要确定通过给任意一个学校发送新软件,这个

P2746 [USACO5.3]校园网Network of Schools tarjan 缩点

题意 给出一个有向图,A任务:求最少需要从几个点送入信息,使得信息可以通过有向图走遍每一个点B任务:求最少需要加入几条边,使得有向图是一个强联通分量 思路 任务A,比较好想,可以通过tarjan缩点,求出入度为0的点的个数任务B一开始以为任务A,B没有关系其实是入度为0的点的个数.出度为0的点的个数的最大值.因为任务B要求在任意学校投放软件使得所有学校都能收到,所以很明显是需要整张图形成一个环,而环中所有节点入度和出度都不为0,所以需要把所有入度和出度的点度数增加. (注意判断本身就全联通的情况

luogu P2746 [USACO5.3]校园网Network of Schools

题目描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软件(称作“接受学校”).注意即使 B 在 A 学校的分发列表中, A 也不一定在 B 学校的列表中. 你要写一个程序计算,根据协议,为了让网络中所有的学校都用上新软件,必须接受新软件副本的最少学校数目(子任务 A).更进一步,我们想要确定通过给任意一个学校发送新软件,这个软件就会分发到网络中的所有学校.为了完成这个任务,我们可能必须扩展接收学校列表,使其加入新成员.计算最少需要增加几个扩展,使得不论我们给

P2746 [USACO5.3]校园网Network of Schools

题目描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软件(称作“接受学校”).注意即使 B 在 A 学校的分发列表中, A 也不一定在 B 学校的列表中. 你要写一个程序计算,根据协议,为了让网络中所有的学校都用上新软件,必须接受新软件副本的最少学校数目(子任务 A).更进一步,我们想要确定通过给任意一个学校发送新软件,这个软件就会分发到网络中的所有学校.为了完成这个任务,我们可能必须扩展接收学校列表,使其加入新成员.计算最少需要增加几个扩展,使得不论我们给

P2746 [USACO5.3]校园网Network of Schools 强连通

题目描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软件(称作“接受学校”).注意即使 B 在 A 学校的分发列表中, A 也不一定在 B 学校的列表中. 你要写一个程序计算,根据协议,为了让网络中所有的学校都用上新软件,必须接受新软件副本的最少学校数目(子任务 A).更进一步,我们想要确定通过给任意一个学校发送新软件,这个软件就会分发到网络中的所有学校.为了完成这个任务,我们可能必须扩展接收学校列表,使其加入新成员.计算最少需要增加几个扩展,使得不论我们给

【luogu2746】 [USACO5.3]校园网Network of Schools [tarjan 缩点]

P2746 [USACO5.3]校园网Network of Schools 任务a:找有多少个入度为0的点 任务b:找出出度为0的个数和入度为0点个数中的较大数 在一个出度为0和另一入度为0的点间连一条边 就可以同时解决两个点 故找出其中较大数 要注意最终缩为一个强连通时要特判 1 #include<bits/stdc++.h> 2 using namespace std; 3 #define ll long long 4 const int N=100+5; 5 int n,in[N],ou

COGS——C 908. 校园网 || 洛谷——P 2746 [USACO5.3]校园网Network of Schools

http://www.cogs.pro/cogs/problem/problem.php?pid=908   ||  https://www.luogu.org/problem/show?pid=2746 ★★   输入文件:schlnet.in   输出文件:schlnet.out   简单对比时间限制:1 s   内存限制:128 MB USACO/schlnet(译 by Felicia Crazy) 描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软

[USACO5.3]校园网Network of Schools

传送门 这道题还是比较容易看出是tarjan的.首先我们知道如果学校之间成环的话那么学校之间一定能到达,直接缩成一个点就好了. 缩完点之后我们得到了一个DAG.之后因为子任务A要求的是最少接受新软件的学校有多少个,可以很容易的想出我们只要给所有入度为0的学校发一份就可以了,因为剩下的必然是可以从其他学校传过来的嘛. 子任务A的答案就是DAG中入度为0的点数. 至于子任务2,我们要求的就是把这个图变为强连通分量至少要加几条边.思考之后发现,一个点如果入度为0,那么它就无法被其他学校传进来,那么就不

luogu2746 [USACO5.3]校园网Network of Schools

题目 题目描述 一些学校连入一个电脑网络.那些学校已订立了协议:每个学校都会给其它的一些学校分发软件(称作“接受学校”).注意即使 B 在 A 学校的分发列表中, A 也不一定在 B 学校的列表中. 你要写一个程序计算,根据协议,为了让网络中所有的学校都用上新软件,必须接受新软件副本的最少学校数目(子任务 A).更进一步,我们想要确定通过给任意一个学校发送新软件,这个软件就会分发到网络中的所有学校.为了完成这个任务,我们可能必须扩展接收学校列表,使其加入新成员.计算最少需要增加几个扩展,使得不论