zoukankan      html  css  js  c++  java
  • [NOI2001]食物链

    [NOI2001]食物链

    这题我采用的是并查集扩展域做法,当然还有边带权的做法,但我还不会.

    我们发现,所有的动物相对于每一种动物都能划分出三个集合 (:) 同类集合,食物集合,天敌集合.

    所以,把每一种动物进行扩展域,用三个并查集维护.

    每次读进一个同类就判断其中一个是否在另一个的食物集合,天敌集合中即可.

    如果矛盾,就统计答案,否则就把两个动物的三个集合一一合并,因为显然同类的天敌和食物都一样.

    读进一个 (x)(y) 的捕食操作呢?

    判断是否存在我吃你而你吃我的情况,这显然是矛盾的.

    另一种矛盾的情况是它们已经是同类了,直接查询即可.

    如果不矛盾,就合并 (x) 的同类集合和 (y) 的天敌集合, (x) 的食物集合和 (y) 的同类集合, (x) 的食物集合和 (y) 的食物集合.

    因为既然 (x)(y) 那么 (x) 的同类显然都吃 (y) ,并且 (y) 的同类显然都是 (x) 的食物,根据题意还可以知道 (y) 的食物也是 (x) 的食物.

    于是这题就做完了.

    扩展域也算是并查集的套路,见过一次基本就会了.

    #include <algorithm>
    #include <iostream>
    #include <cstdlib>
    #include <cstring>
    #include <cstdio>
    #include <string>
    #include <vector>
    #include <queue>
    #include <cmath>
    #include <ctime>
    #include <map>
    #include <set>
    #define MEM(x,y) memset ( x , y , sizeof ( x ) )
    #define rep(i,a,b) for (int i = (a) ; i <= (b) ; ++ i)
    #define per(i,a,b) for (int i = (a) ; i >= (b) ; -- i)
    #define pii pair < int , int >
    #define one first
    #define two second
    #define rint read<int>
    #define pb push_back
    #define db double
    #define ull unsigned long long
    #define lowbit(x) ( x & ( - x ) )
    
    using std::queue ;
    using std::set ;
    using std::pair ;
    using std::max ;
    using std::min ;
    using std::priority_queue ;
    using std::vector ;
    using std::swap ;
    using std::sort ;
    using std::unique ;
    using std::greater ;
    
    template < class T >
        inline T read () {
            T x = 0 , f = 1 ; char ch = getchar () ;
            while ( ch < '0' || ch > '9' ) {
                if ( ch == '-' ) f = - 1 ;
                ch = getchar () ;
            }
           while ( ch >= '0' && ch <= '9' ) {
                x = ( x << 3 ) + ( x << 1 ) + ( ch - 48 ) ;
                ch = getchar () ;
           }
           return f * x ;
        }
    
    const int N = 5e4 + 100 ;
    
    int n , k , f[N*3] , ans ; // i is self , i + n is enemy , i + 2n is eaten ones.
    
    inline int getf (int x) { return f[x] == x ? x : f[x] = getf ( f[x] ) ; }
    
    inline void merge (int x , int y) {
        x = getf ( x ) ; y = getf ( y ) ;
        if ( x != y ) f[y] = x ; return ;
    }
    
    int main (int argc , char * argv[]) {
        n = rint () ; k = rint () ;
        rep ( i , 1 , n * 3 ) f[i] = i ;
        while ( k -- ) {
            int opt = rint () , u = rint () , v = rint () ;
            if ( u < 1 || u > n || v < 1 || v > n ) { ++ ans ; continue ; }
            if ( opt == 1 ) {
                int tx = getf ( u ) , ty = getf ( v ) ;
                if ( tx == getf ( v + n ) || tx == getf ( v + 2 * n ) ) { ++ ans ; continue ; }
                if ( ty == getf ( u + n ) || ty == getf ( u + 2 * n ) ) { ++ ans ; continue ; }
                merge ( u , v ) ; merge ( u + n , v + n ) ; merge ( u + 2 * n , v + 2 * n ) ;
            } else {
                if ( u == v ) { ++ ans ; continue ; }
                int tx = getf ( u ) , ty = getf ( v ) ;
                if ( tx == ty ) { ++ ans ; continue ; }
                if ( getf ( u + n ) == ty ) { ++ ans ; continue ; }
                if ( tx == getf ( v + 2 * n ) ) { ++ ans ; continue ; }
                merge ( u , v + n ) ; merge ( u + 2 * n , v ) ; merge ( u + n , v + 2 * n ) ;
            }
        }
        printf ("%d
    " , ans ) ;
        #ifndef ONLINE_JUDGE
        system ("pause") ;
        #endif
        return 0 ;
    }
    
  • 相关阅读:
    高仿富途牛牛-组件化(三)-界面美化
    高仿富途牛牛-组件化(二)-磁力吸附
    高仿富途牛牛-组件化(一)-支持页签拖拽、增删、小工具
    Cef3 学习资料
    QCustomplot使用分享(八) 绘制图表-加载cvs文件
    Electron桌面应用:环境搭建
    Qt疑难问题-模态窗口父类被析构
    Qt线程实现分析-moveToThread vs 继承
    基础连接已经关闭: 未能为 SSL/TLS 安全通道建立信任关系。
    asp.net中的ListBox控件添加双击事件
  • 原文地址:https://www.cnblogs.com/Equinox-Flower/p/11773780.html
Copyright © 2011-2022 走看看