一、题目链接

  http://acm.hdu.edu.cn/showproblem.php?pid=5222

二、题意

  给一个无向边+有向边的混合图,其中每条边只能使用一次,问图中是否存在环。

三、思路

  1、必备知识点:图中判断是否存在环

  图是否连通都没关系。
  • 无向图
  (1)DFS
  (2)BFS
  (3)并查集
  • 有向图
  (1)拓扑排序
  • 无向边+有向边的混合图
  (1)法1
    ①把所有无向边(u, v)加入到并查集,不加入图中;
    ②对于所有有向边(u, v),在并查集中找出u和v的根结点x和y,图中添加x到y的边(本质就是把并查集中所有的点合并成一个点);
    ③对建立的图做和【有向图】相同的拓扑排序;
    另外,在①和②中,在并查集合并结点u和v的时候,如果u和v已经属于同一个组(祖先相同),那么,说明已经存在环了。
 
   有了上面的知识点,这个题就好做了。直接套一下方法就OK。
  PS:做这个题时,一直纠结如何用一种优雅的方式浓缩并查集中的所有点,思来想去,觉得这个“一边输入,一边浓缩点”的方法,确实是,棒!
四、源代码
#pragma comment(linker, "/STACK:102400000,102400000")
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<set>
#include<vector>
#include<queue>
using namespace std;
#define MAXN 1000100
typedef long long LL;
typedef struct{
    int par[MAXN], rk[MAXN];
    void init(){
        ;i < MAXN;++i)par[i] = i, rk[i] = ;
    }
    int root(int x){
        return x == par[x] ? x : par[x] = root(par[x]);
    }
    void unite(int u, int v){
        int x = root(u), y = root(v);
        if(x == y)return;
        if(rk[x] < rk[y])par[x] = y;
        else{
            par[y] = x;
            if(rk[x] == rk[y])rk[y]++;
        }
    }
    bool same(int u, int v){
        return root(u) == root(v);
    }
}UniFinder;

int N, M1, M2;
vector<int> g[MAXN];
int deg[MAXN];

template<class T> inline void read(T& x){
    char t;
    bool sign = false;
    '));
    if(t == '-')sign = true, t =getchar();
    x = t - ';
     + t - ';
    if(sign)x = -x;
}

priority_queue<int, vector<int>, greater<int> > que;
bool topo_sort(){
    while(!que.empty())que.pop();
    ;i <= N;++i){
        )que.push(i);
    }
    int rest = N;
    while(!que.empty()){
        int top = que.top();
        que.pop();
        rest--;
        , sz = g[top].size();i < sz;++i){
            int to = g[top][i];
            deg[to]--;
            )que.push(to);
        }
    }
    ;
}

UniFinder uf;
int main(){
#ifndef ONLINE_JUDGE
    freopen("input.txt", "r", stdin);
#endif
    int T, u, v;
    scanf("%d", &T);
    while(T--){
        ;i < MAXN;++i)g[i].clear();
        memset(deg, , sizeof(deg));
        uf.init();
        bool ans = false;
        scanf("%d%d%d", &N, &M1, &M2);
        ;i < M1;++i){
            read(u), read(v);
            if(uf.same(u, v))ans = true;
            uf.unite(u, v);
        }
        ;i < M2;++i){
            read(u), read(v);
            u = uf.root(u), v = uf.root(v);
            if(u == v)ans = true;
            if(ans)continue;
            g[u].push_back(v), deg[v]++;
        }
        if(ans)puts("YES");
        else printf("%s\n", topo_sort() ? "YES" : "NO");
    }
    ;
} 

HDU-5222 Exploration(拓扑排序)的更多相关文章

  1. HDU.2647 Reward(拓扑排序 TopSort)

    HDU.2647 Reward(拓扑排序 TopSort) 题意分析 裸的拓扑排序 详解请移步 算法学习 拓扑排序(TopSort) 这道题有一点变化是要求计算最后的金钱数.最少金钱值是888,最少的 ...

  2. HDU 5222 ——Exploration——————【并查集+拓扑排序判有向环】

    Exploration Time Limit: 30000/15000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)T ...

  3. ACM: hdu 2647 Reward -拓扑排序

    hdu 2647 Reward Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Des ...

  4. HDU 2647 Reward (拓扑排序)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2647 题意是给你n点m条有向边,叶子点(出度为0)上的值为888,父亲点为888+1,依次计算... ...

  5. hdu 5438 Ponds 拓扑排序

    Ponds Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/contests/contest_showproblem ...

  6. hdu 4857 逃生 拓扑排序+PQ,剥层分析

    pid=4857">hdu4857 逃生 题目是求拓扑排序,但不是依照字典序最小输出,而是要使较小的数排在最前面. 一開始的错误思路:给每一个点确定一个优先级(该点所能到达的最小的点) ...

  7. HDU 1285 经典拓扑排序入门题

    确定比赛名次 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  8. HDU 2647 逆向拓扑排序

    令每一个员工都有一个自己的等级level[i] , 员工等级越高,那么工资越高,为了使发的钱尽可能少,所以每一级只增加一单位的钱 输入a b表示a等级高于b,那么我们反向添加边,令b—>a那么i ...

  9. HDU 4917 Permutation 拓扑排序的计数

    题意: 一个有n个数的排列,给你一些位置上数字的大小关系.求合法的排列有多少种. 思路: 数字的大小关系可以看做是一条有向边,这样以每个位置当点,就可以把整个排列当做一张有向图.而且题目保证有解,所以 ...

随机推荐

  1. Observer(观察者)

    意图: 定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时, 所有依赖于它的对象都得到通知并被自动更新. 适用性: 当一个抽象模型有两个方面, 其中一个方面依赖于另一方面.将这二者封装在独立 ...

  2. Tensorflow一些常用基本概念与函数(二)

    1.tensorflow的基本运作 为了快速的熟悉TensorFlow编程,下面从一段简单的代码开始: import tensorflow as tf #定义‘符号’变量,也称为占位符 a = tf. ...

  3. [Mybatis]resultMap的使用总结

    resultMap是Mybatis最强大的元素,它可以将查询到的复杂数据(比如查询到几个表中数据)映射到一个结果集当中. resultMap包含的元素的相关属性解释如下: <!--column不 ...

  4. windows下的IO模型之异步选择(WSAAsyncSelect)模型

    异步选择(WSAAsyncSelect)模型是一个有用的异步I/O 模型.其核心函数是WSAAsyncSelect,该函数是非阻塞的 (关于异步io的理解详情可以看:http://www.cnblog ...

  5. UVA-1601 The Morning after Halloween(BFS或双向BFS)

    题目大意:在一张图中,以最少的步数将a,b,c移到对应的A,B,C上去.其中,每个2x2的方格都有障碍并且不能两个小写字母同时占据一个格子. 题目分析:为避免超时,先将图中所有能联通的空格建起一张图, ...

  6. JavaScript学习总结(二十一)——使用JavaScript的数组实现数据结构中的队列与堆栈

    今天在项目中要使用JavaScript实现数据结构中的队列和堆栈,这里做一下总结. 一.队列和堆栈的简单介绍 1.1.队列的基本概念 队列:是一种支持先进先出(FIFO)的集合,即先被插入的数据,先被 ...

  7. LINUX中的RCU机制的分析

    RCU机制是Linux2.6之后提供的一种数据一致性访问的机制,从RCU(read-copy-update)的名称上看,我们就能对他的实现机制有一个大概的了解,在修改数据的时候,首先需要读取数据,然后 ...

  8. cvs的规范以及介绍(转)

    原文链接:http://blog.csdn.net/zlzlei/article/details/9236403 1. CSV的全称是叫Comma Separated Value 2. CSV的MIM ...

  9. timer Compliant Controller project (4)layout and gerber, paning

    1 LAYOUT 2 Gerber 3 CAM350-Paining

  10. 关于C#引用dll动态链接库文件的注释问题

    1.dll动态库文件项目生成属性中要勾选"XML文档文件" 注意:1).要选中项目,查看项目属性,选中解决方案是找不到的.2).XML文件的名字不要修改. 2.添加引用时XML文件 ...