题解 P2002 消息扩散

题解 P2002 消息扩散

题目链接

非常裸的一道缩点题,很显然每个强连通分量里给一个点消息就够了,缩完点以后很显然我们要讨论每一个点的入度。对于入度为零的点(没有其它点能给他消息),我们就必须给它一份消息,所以就变成了数入度为零的点。由于数据规模,所以我们不需要重新建图。

值得一提的是一个小细节。重边和自环。我一开始用map判重边结果t了一个点(也有可能是这个点有自环qaq),然后删了map特判判自环就直接过了,看来重边好像对这道题没什么影响??(大雾

代码:几乎是裸的模版题,好像比模版题还简单点

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <stack>
#include <vector>
#include <map>
#define N 100005
#define ll long long
using namespace std;

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;
}
vector < int > a[N];
stack < int > s;
map < pair < int, int >, int > mp;
int n, m, times, dfn[N], low[N], instc[N], vis[N], mtot, du[N], ans;
//vector存图可能会慢
void tarjan(int x)
{
    dfn[x] = low[x] = ++times;
    instc[x] = 1;
    s.push(x);
    for (int i = 0; i < a[x].size(); i++)
    {
        int u = a[x][i];
        if (!dfn[u])
        {
            tarjan(u);
            low[x] = min(low[x], low[u]);
        }
        else if (instc[u])
            low[x] = min(low[x], dfn[u]);
    }
    if (dfn[x] == low[x])
    {
        ++mtot;
        vis[x] = mtot;
        instc[x] = false;
        while (s.top() != x)
        {
            vis[s.top()] = mtot;
            instc[s.top()] = 0;
            s.pop();
        }
        s.pop();
    }
}

int main()
{
    n = read(), m = read();
    for (int i = 1; i <= m; i++) { int a1 = read(), a2 = read(); if (a1 != a2) a[a1].push_back(a2); }
    for (int i = 1; i <= n; i++)
        if (!dfn[i]) tarjan(i);
    for (int i = 1; i <= n; i++)
    {
        for (int j = 0; j < a[i].size(); j++)
        {
            int u = a[i][j];
            if (vis[u] != vis[i])
            {
                du[vis[u]]++; //更新入度 (不在乎是否重复)
            }
        }
    }
    for (int i = 1; i <= mtot; i++)
        if (du[i] == 0) ans++; //统计入度累加答案
    cout << ans << endl;
    return 0;
}

最后推荐几个相似题目,做tarjan必刷:

P1262 间谍网络

P2341 [HAOI2006]受欢迎的牛|【模板】强连通分量

P3627 [APIO2009]抢掠计划

最后吐槽下为什么这么裸的tarjan都是蓝题,因为普及组不考么qaq

19.09.05

原文地址:https://www.cnblogs.com/YuanqiQHFZ/p/11622349.html

时间: 2024-10-16 16:50:25

题解 P2002 消息扩散的相关文章

P2002 消息扩散

P2002 消息扩散 题目背景 本场比赛第一题,给个简单的吧,这 100 分先拿着. 题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入输出格式 输入格式: 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式: 一行一个整数,表示至少要在几个城市中发布消息. 输入输出样例 输入样例#1: 5 4 1

洛谷P2002 消息扩散

题目背景 本场比赛第一题,给个简单的吧,这 100 分先拿着. 题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入输出格式 输入格式: 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式: 一行一个整数,表示至少要在几个城市中发布消息. 输入输出样例 输入样例#1: 5 4 1 2 2 1 2 3 5

luogu P2002 消息扩散

题目背景 本场比赛第一题,给个简单的吧,这 100 分先拿着. 题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入输出格式 输入格式: 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式: 一行一个整数,表示至少要在几个城市中发布消息. 输入输出样例 输入样例#1: 5 4 1 2 2 1 2 3 5

洛谷—— P2002 消息扩散

https://www.luogu.org/problem/show?pid=2002 题目背景 本场比赛第一题,给个简单的吧,这 100 分先拿着. 题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入输出格式 输入格式: 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式: 一行一个整数,表示至少要

洛谷 P2002 消息扩散

题目背景 本场比赛第一题,给个简单的吧,这 100 分先拿着. 题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入输出格式 输入格式: 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式: 一行一个整数,表示至少要在几个城市中发布消息. 输入输出样例 输入样例#1: 5 4 1 2 2 1 2 3 5

P2002 消息扩散[SCC缩点]

题目描述 有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息. 输入格式 第一行两个整数n,m表示n个城市,m条单向道路. 以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环. 输出格式 一行一个整数,表示至少要在几个城市中发布消息. 输入输出样例 输入 #1复制 输出 #1复制 说明/提示 [数据范围] 对于20%的数据,n≤200; 对于40%的数据,n≤2,000; 对于1

【Luogu P2002&amp;P2341】消息扩散/受欢迎的奶牛

Luogu P2002 Luogu P2341 使用强连通分量算法缩点 第一题统计入度为0的个数强连通分量数. 第二题的答案为当且仅当仅有一个强连通分量的出度为0时该强连通分量的节点数,原因如下:若一个强连通分量出度为0,则说明这个强连通分量的喜爱无法对外传递:若有多个强连通分量出度为0,则说明这几个强连通分量的喜爱无法对外传递,则不可能产生明星. 有关强连通分量的知识,可以看我的另一篇博客 [Luogu P3387]缩点模板(强连通分量Tarjan&拓扑排序) AC代码 P2002: #inc

【luogu P2002】消息扩散

https://www.luogu.org/problem/show?pid=2002 SCC缩点的模板题,缩点后统计入度为0的点的数量就完了. #include <iostream> #include <vector> #include <stack> #include <cstring> #define maxn 100005 using namespace std; int n, m; vector<int> g[maxn], g2[max

强连通分量——消息扩散(洛谷_2002)——tarjan求scc

强连通分量(scc) 缩点 建新图 找入度为0的点 大功告成 #include<iostream> #include<cstdio> #include<algorithm> #include<cstring> #include<vector> #include<stack> using namespace std; inline int read(){ int t=1,num=0;char c=getchar(); while(c&g