cogs 1001. [WZOI2011 S3] 消息传递 Tarjan

1001. [WZOI2011 S3] 消息传递

★★   输入文件:messagew.in   输出文件:messagew.out   简单对比
时间限制:1 s   内存限制:128 MB

Problem 2 消息传递 (messagew.pas/c/cpp)
问题描述
WZland开办了一个俱乐部(这里面可以干任何的事情),这引来了许多的人来加入。俱乐部的人数越来越多,关系也越来越复杂……
俱乐部的人来自各个地方,为了增加友谊,俱乐部举行了一次晚会。晚会上又进行了一个传话游戏,如果A认识B,那么A收到某个消息,就会把这个消息传给B,以及所有A认识的人(如果A认识B,B不一定认识A),所有人从1到N编号。
现在给出所有“认识”关系,俱乐部的负责人WZland的国王想知道一个十分简单的问题:如果A发布一条新消息,那么会不会经过若干次传话后,这个消息传回给了A,1≤A≤N。但是WZland的国王是出了名的数学差,幸好的是你在他的身边,于是他就将这个问题交给你来解决。
输入格式
输入数据中的第一行是两个数N和M,两数之间有一个空格,表示人数和认识关系数。
接下来的M行,每行两个数A和B,表示A认识B(1A, BN,AB)。
输出格式
输出文件中一共有N行,每行一个字符“T”或“F”。第i行如果是“T”,表示i发出一条新消息会传回给i;如果是“F”,表示i发出一条新消息不会传回给i。
样例输入输出
messagwe.in 
4 6
1 2
2 3
4 1
3 1
1 3
2 3
messagew.out
T
T
T
F
数据规模
对于30%的数据,N≤1000,M≤20000;
对于50%的数据,N≤10000,M≤100000;
对于100%的数据,N≤100000,M≤200000;
认识关系可能会重复给出。
时间限制
1s

这一道题其实还是比较简单的 就是一个Tarjan而已
那么应该怎么进行处理呢??只要  当前点所在的强连通分量的大小>1 就相当于 这个点传出去以后 还能有人再给他传回来!else  就是他自己一个人 肯定不行啦

至于那个认识关系可能会重复。。。我一开始也没有想到怎样处理 就闭着眼交了上去 居然A了 后来想明白了(还是板子高明啊)快看快看:

那个第67行看到没  就是如果to点已经在栈中了  就把当前点的low 和 to点的dfn取一个较小的值 直接就返回了 不会出bug了

(<<本人心声:清华orz的板子就是厉害!赞一个)

下面来贴出简洁清晰(自认为)的代码吧

#include<vector>
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cmath>
#define maxn 100005
using namespace std;
int n,m;
vector<int> v[maxn];
int dfn[maxn],low[maxn],size[maxn],scc_cnt,st[maxn],cnt,tim,belong[maxn];
bool bein[maxn];
void Tarjan(int rt)
{
    dfn[rt]=low[rt]=++tim;//时间戳++ 初始化dfn和low相等
    st[++cnt]=rt;//该点入栈
    bein[rt]=true; //标记是否在栈中的数组
    for(int i=0;i<v[rt].size();i++)
    {
        int to=v[rt][i];
        if(!dfn[to])
            Tarjan(to),low[rt]=min(low[rt],low[to]);
        else if(bein[to])
            low[rt]=min(low[rt],dfn[to]);
    }
    if(dfn[rt]==low[rt])
    {
        scc_cnt++;//统计强连通分量
        int k;
        do{
            k=st[cnt];
            cnt--;
            belong[k]=scc_cnt;
            bein[k]=false;
            size[scc_cnt]++;
        }while(k^rt);//^ 在这里就相当于!=        do while  是先跑一遍再判断
    }
}
int main()
{
    freopen("messagew.in","r",stdin);
    freopen("messagew.out","w",stdout);
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++)
    {
        int x,y;scanf("%d%d",&x,&y);
        v[x].push_back(y);//这里要注意是一条单项边啊
    }
    for(int i=1;i<=n;i++)
        if(!dfn[i])//如果未访问
            Tarjan(i);
    for(int i=1;i<=n;i++)
    {
        if(size[belong[i]]>1)
            puts("T");
        else
            puts("F");
    }
    return 0;
}

原文地址:https://www.cnblogs.com/Tidoblogs/p/11329587.html

时间: 2024-11-15 13:14:46

cogs 1001. [WZOI2011 S3] 消息传递 Tarjan的相关文章

cogs——1001. [WZOI2011 S3] 消息传递

1001. [WZOI2011 S3] 消息传递 ★★   输入文件:messagew.in   输出文件:messagew.out   简单对比时间限制:1 s   内存限制:128 MB Problem 2 消息传递 (messagew.pas/c/cpp) 问题描述 WZland开办了一个俱乐部(这里面可以干任何的事情),这引来了许多的人来加入.俱乐部的人数越来越多,关系也越来越复杂…… 俱乐部的人来自各个地方,为了增加友谊,俱乐部举行了一次晚会.晚会上又进行了一个传话游戏,如果A认识B,

cojs 1001. [WZOI2011 S3] 消息传递

1001. [WZOI2011 S3] 消息传递 ★★   输入文件:messagew.in   输出文件:messagew.out   简单对比时间限制:1 s   内存限制:128 MB Problem 2 消息传递 (messagew.pas/c/cpp) 问题描述 WZland开办了一个俱乐部(这里面可以干任何的事情),这引来了许多的人来加入.俱乐部的人数越来越多,关系也越来越复杂…… 俱乐部的人来自各个地方,为了增加友谊,俱乐部举行了一次晚会.晚会上又进行了一个传话游戏,如果A认识B,

1002. [WZOI2011 S3] 周年纪念日

1002. [WZOI2011 S3] 周年纪念日 Problem 3 周年纪念日 (anniversary.pas/c/cpp) 背景 WZland即将迎来一个举国欢庆的日子—建国150亿周年纪念日,值此之际WZland有许多事情要准备…… 问题描述 WZland的国王决定举办一个晚会,这次晚会要求所有的WZland居民都参加,但是他发现WZland的所有城市之间的道路都已经毁坏(平时WZland的居民都在自己的城市里活动,所以他们对于那些道路一点都不关心).这是一件十分麻烦的事情,因为这个晚

求图的强连通分量--tarjan算法

一:tarjan算法详解 ?思想: ? ?做一遍DFS,用dfn[i]表示编号为i的节点在DFS过程中的访问序号(也可以叫做开始时间)用low[i]表示i节点DFS过程中i的下方节点所能到达的开始时间最早的节点的开始时间.(也就是之后的深搜所能到达的最小开始时间)初始时dfn[i]=low[i] ? ?在DFS过程中会形成一搜索树.在搜索树上越先遍历到的节点,显然dfn的值就越小. ? ?DFS过程中,碰到哪个节点,就将哪个节点入栈.栈中节点只有在其所属的强连通分量已经全部求出时,才会出栈. ?

C++语言学习(四)——类与对象

C++语言学习(四)--类与对象 一.构造函数(constructor) 1.构造函数简介 C++语言中,构造函数是与类名相同的特殊成员函数.在类对象创建时,自动调用构造函数,完成类对象的初始化.类对象本身是变量,在栈.堆上创建的对象,对象的成员初始化为随机值:在静态存储区创建的对象,对象的成员初始化为0. 2.构造函数的定义 构造函数声明的语法如下:classname(parameters);没有参数的构造函数称为无参构造函数.当类中没有定义构造函数(包括拷贝构造函数)时,编译器默认提供一个无

cogs 942. [東方S3] 比那名居天子

二次联通门 : cogs 942. [東方S3] 比那名居天子 /* cogs 942. [東方S3] 比那名居天子 二分水题 二分所求区间长度 判断后缩小范围即可 */ #include <cstring> #include <cstdio> #define Max 1000000 void read (int &now) { now = 0; register char word = getchar (); while (word < '0' || word &g

cogs 944. [東方S3] 藤原妹红

二次联通门 : cogs 944. [東方S3] 藤原妹红 /* cogs 944. [東方S3] 藤原妹红 最小生成树 + 树形dp 首先对原图跑最下生成树 后建出一棵树 在树上进行dp 先走到叶子节点, 顺便处理出距离 最终回溯时更新答案 */ #include <algorithm> #include <cstdio> #define INF 1e30 void read (int &now) { register char word = getchar (); fo

tarjan——cogs 1298 通讯问题

1298. 通讯问题 ★   输入文件:jdltt.in   输出文件:jdltt.out   简单对比 时间限制:1 s   内存限制:128 MB [题目描述] 一个篮球队有n个篮球队员,每个队员都有联系方式(如电话.电子邮件等).但并不是每个队员的联系方式都公开,每个队员的联系方式只有一部分队员知道.问队员可以分成多少个小组,小组成员之间可以相互通知(包括一个队员一个组,表示自己通知自己). [输入格式] 输入文件有若干行 第一行,一个整数n,表示共有n个队员(2<=n<=100) 下面

COGS——T 8. 备用交换机

http://www.cogs.pro/cogs/problem/problem.php?pid=8 ★★   输入文件:gd.in   输出文件:gd.out   简单对比时间限制:1 s   内存限制:128 MB [问题描述] n个城市之间有通讯网络,每个城市都有通讯交换机,直接或间接与其它城市连接.因电子设备容易损坏,需给通讯点配备备用交换机.但备用交换机数量有限,不能全部配备,只能给部分重要城市配置.于是规定:如果某个城市由于交换机损坏,不仅本城市通讯中断,还造成其它城市通讯中断,则配