zoukankan      html  css  js  c++  java
  • [题解] [JSOI2013] 侦探jyy

    题面

    题解

    看哪些事件不发生也对答案没有影响

    假如说这件事不发生, 那么他的前驱都不会发生

    然后把所有入度为零并且不是这件事的前驱的点全部设为发生, BFS 一遍, 看那 M 个是否都能发生

    若能, 则该事件不一定发生

    若不能, 该事件一定发生

    复杂度 (O(nm))

    Code

    #include <algorithm>
    #include <iostream>
    #include <cstring>
    #include <cstdio>
    #include <queue>
    const int N = 100005; 
    using namespace std;
    
    int n, m, t, head[N], head1[N], cnt, tot, d[N], vis[N], hp[N], ck[N], ans[N];
    struct edge { int to, nxt; } e[N], E[N]; 
    queue<int> q; 
    
    template < typename T >
    inline T read()
    {
        T x = 0, w = 1; char c = getchar();
        while(c < '0' || c > '9') { if(c == '-') w = -1; c = getchar(); }
        while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
        return x * w; 
    }
    
    inline void adde(int u, int v)
    {
        e[++cnt] = (edge) { v, head[u] }, head[u] = cnt;
        E[++tot] = (edge) { u, head1[v] }, head1[v] = tot; 
    }
    
    bool check(int x)
    {
        q.push(x), vis[x] = ++cnt;
        while(!q.empty())
        {
    	int u = q.front(); q.pop();
    	for(int v, i = head1[u]; i; i = E[i].nxt)
    	    if(vis[v = E[i].to] != cnt) vis[v] = cnt, q.push(v); 
        }
        for(int i = 1; i <= t; i++)
    	if(vis[hp[i]] == cnt) return 1;
        for(int i = 1; i <= n; i++)
    	if(!d[i] && vis[i] != cnt) q.push(i), vis[i] = cnt;
        while(!q.empty())
        {
    	int u = q.front(); q.pop();
    	for(int v, i = head[u]; i; i = e[i].nxt)
    	    if(vis[v = e[i].to] != cnt) vis[v] = cnt, q.push(v); 
        }
        for(int i = 1; i <= t; i++)
    	if(vis[hp[i]] != cnt) return 1;
        return 0; 
    }
    
    int main()
    {
        n = read <int> (), m = read <int> (), t = read <int> ();
        for(int u, v, i = 1; i <= m; i++)
        {
    	u = read <int> (), v = read <int> ();
    	adde(u, v), d[v]++; 
        }
        cnt = 0; 
        for(int i = 1; i <= t; i++)
    	ck[hp[i] = read <int> ()] = 1;
        for(int i = 1; i <= n; i++)
    	if(ck[i] || check(i)) ans[++ans[0]] = i;
        for(int i = 1; i <= ans[0]; i++)
    	printf("%d%c", ans[i], i == ans[0] ? '
    ' : ' '); 
        return 0; 
    }
    
  • 相关阅读:
    sql导数据 自增长
    只能在执行 Render() 的过程中调用 RegisterForEventValidation
    JS获取DropDownList的值
    解决IE6、IE7、IE8、Firefox兼容的两种方案
    C#日期格式化
    页面弹出窗口刷新父页面方式小结
    Dictionary Queue Stack SortedList ArrayList
    asp.net水晶报表推模式加载报表数据代码段
    JS隐藏工具栏菜单栏
    解决在SQL Server 2000的存储过程不能调试
  • 原文地址:https://www.cnblogs.com/ztlztl/p/12292555.html
Copyright © 2011-2022 走看看