Fork me on GitHub

[NOI2001]食物链

[NOI2001]食物链

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

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

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

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

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

读进一个 $x$ 吃 $y$ 的捕食操作呢?

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

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

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

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

于是这题就做完了.

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#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\n" , ans ) ;
#ifndef ONLINE_JUDGE
system ("pause") ;
#endif
return 0 ;
}