zoukankan      html  css  js  c++  java
  • 【题解】Atcoder AGC#16 E-Poor Turkeys

      %拜!颜神怒A此题,像我这样的渣渣只能看看题解度日╭(╯^╰)╮在这里把两种做法都记录一下吧~

      题解做法:可以考虑单独的一只鸡 u 能否存活。首先我们将 u 加入到集合S。然后我们按照时间倒序往回推,如果在时间 t 的时候发现有 u 和 v 同时被抉择,为了保证 u 的存活我们只能杀掉 v,也就是说在 t - 1的时刻 v 必须存活。这时我们将 v 加入到集合 S 中,再继续进行这个过程。如果在某个时刻我们发现 u 和 v 同时被抉择,可 u 和 v 都已经在集合中出现过了(要求在这个时刻一并存活),这样显然是非法的。所以可以判定 u 没有存活的可能。

      如果一只鸡 u 能够存活,我们把这个过程中获得的 S 集合称作 (S_{u}) 。u 和 v 能够共存的充要条件即为 u 和 v 均有存活的可能,且 (S_{u}) 和 (S_{v}) 两个集合不存在交集。为什么呢?因为一只鸡在 t 时刻出现在了 S 集合中,说明它将在 t 时刻被杀掉。如果两个集合中 x 出现的时间不同,那么出现了冲突;但它们又不可能在同一个时间出现,因为一个时间节点只有唯一的一个抉择,反推回去也必然都是一样的,但开始的节点一个是 u,一个是 v,所以不可能。得证。

      代码:

    #include <bits/stdc++.h>
    using namespace std;
    #define maxn 405
    #define maxm 100500
    int n, m, x[maxm], y[maxm], mark[maxn];
    int ans, S[maxn][maxn];
    
    int read()
    {
        int x = 0, k = 1;
        char c; c = getchar();
        while(c < '0' || c > '9') { if(c == '-') k = -1; c = getchar(); }
        while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
        return x * k;
    }
    
    int main()
    {
        n = read(), m = read();
        for(int i = 1; i <= m; i ++) x[i] = read(), y[i] = read();
        for(int i = 1; i <= n; i ++)
        {
            memset(mark, 0, sizeof(mark)); 
            mark[i] = 1; S[i][++ S[i][0]] = i;
            for(int j = m; j >= 1; j --)
            {
                if(mark[x[j]] && mark[y[j]]) { S[i][0] = -1; break; }
                if(mark[x[j]]) mark[y[j]] = 1;
                else if(mark[y[j]]) mark[x[j]] = 1;
            }
            if(S[i][0] == -1) continue;
            for(int j = 1; j <= n; j ++)
                if(mark[j]) S[i][0] ++, S[i][S[i][0]] = j;
        }
        
        for(int i = 1; i <= n; i ++)
            for(int j = i + 1; j <= n; j ++)
            {
                if(S[i][0] == -1 || S[j][0] == -1) continue;
                memset(mark, 0, sizeof(mark)); bool flag = 1;
                for(int k = 1; k <= S[i][0]; k ++) mark[S[i][k]] = 1;
                for(int k = 1; k <= S[j][0]; k ++) 
                    if(mark[S[j][k]]) { flag = 0; break; }
                if(flag) ans ++;
            }
        printf("%d
    ", ans);
        return 0;
    }

      下面是颜神的解法(并没有代码...)也非常的妙,而且复杂度比题解还低……可以考虑建出一张图,在这张图上面所有有连边的鸡均无法共存来获得答案。那么如何建出这张图?一个人选择了 u 和 v 这两只鸡,那么这两只鸡是一定不可能共存的。假设我们在 t - 1 时刻建出的图满足在 t - 1 时刻及之前出现的所有抉择所限制不能共存的鸡均有连边,那么考虑加入t时刻的抉择之后会对这张图产生什么影响。

      考虑 u 和 v 的抉择,会使哪些原本可以和 u 共存的鸡不能再和 u 共存?

      如果图中的一只鸡 x 与 u 没有连边,也与 v 没有连边,那么它与 u 的生死无关;

      如果一只鸡 x 与 v 有连边,而与 u 没有连边,说明 u 和 x 不能共存,我们添加一条从 u 到 x 的边。因为 x 与 v 不能共存,所以若 x 存活,v 一定死亡。那么新的 u,v 边一定会导致 u 的死亡;若 x 死亡,那么 u 可能依然存活,也可能已经死亡;但在这两种情况下,x 都不能与 u 共存。对于 v 我们也是一样的添边。

      最后检查一下哪些节点是可以共存的即可。

  • 相关阅读:
    Python 生成器
    Python 装饰器
    Go语言【第十四篇】:Go语言基础总结
    Go语言【第十三篇】:Go语言递归函数
    Go语言【第十二篇】:Go数据结构之:切片(Slice)、范围(Range)、集合(Map)
    Go语言【第十一篇】:Go数据结构之:结构体
    Java入门之:对象和类
    Alpha阶段第2周/共2周 Scrum立会报告+燃尽图 04
    Alpha阶段第2周/共2周 Scrum立会报告+燃尽图 03
    Alpha阶段第2周/共2周 Scrum立会报告+燃尽图 02
  • 原文地址:https://www.cnblogs.com/twilight-sx/p/9901626.html
Copyright © 2011-2022 走看看