BZOJ 1854 [SCOI2010] 游戏 并查集

题目描述 Description
lxhgww最近迷上了一款游戏,在游戏里,他拥有很多的装备,每种装备都有2个属性,这些属性的值用[1,10000]之间的数表示.当他使用某种装备时,他只能使用该装备的某一个属性.并且每种装备最多只能使用一次.
游戏进行到最后,lxhgww遇到了终极boss,这个终极boss很奇怪,攻击他的装备所使用的属性值必须从1开始连续递增地攻击,才能对boss产生伤害.也就是说一开始的时候,lxhgww只能使用某个属性值为1的装备攻击boss,然后只能使用某个属性值为2的装备攻击boss,然后只能使用某个属性值为3的装备攻击boss……以此类推
现在lxhgww想知道他最多能连续攻击boss多少次?

输入描述 Input Description
输入的第一行是一个整数N,表示lxhgww拥有N种装备
接下来N行,是对这N种装备的描述,每行2个数字,表示第i种装备的2个属性值

输出描述 Output Description
输出一行,包括1个数字,表示lxhgww最多能连续攻击的次数

样例输入 Sample Input
3
1 2
3 2
4 5

样例输出 Sample Output
2

数据范围及提示 Data Size & Hint
对于 30% 30 % 的数据,保证 N1000 N ≤ 1000
对于 100% 100 % 的数据,保证 N1000000 N ≤ 1000000

Solution

这个题目有两种方法都是可做的

  1. 二分图匹配

这个就很显然啦
直接将权值和点连边,按顺序刺激 110000 1 − 10000 的点看是否能形成增广路,每个点最多刺激一次

  1. 并查集

刚开始知道能用并查集做我是懵逼的
看过HZWERdalao的博客后明白了
将[a,b]的一个物品看做连接[a,b]的一条边
用并查集维护集合
若一个集合中能够形成环,则说明集合中的数均可选
若一个集合中没有环,仅仅为一棵树,则说明集合中有一个数选不了
当然我们将集合中最大的数设为选不了就好了

最后从 110000 1 − 10000 按顺序看每个点所处集合选不了的点是不是他本身即可

代码如下:

#include 
using namespace std;
const int N = 1000010;
int point[N],fa[N],n;
int read() {
    int ans=0,flag=1;
    char ch=getchar();
    while( (ch>'9' || ch<'0') && ch!='-' ) ch=getchar();
    if(ch=='-') {flag=-1;ch=getchar();}
    while(ch>='0' && ch<='9') {ans=ans*10+ch-'0';ch=getchar();}
    return ans*flag;
}
int find(int x) {return fa[x]==x?fa[x]:fa[x]=find(fa[x]);}
int main() {
    n=read();
    for(int i=1;i<=10000;++i) point[i]=fa[i]=i;
    for(int i=1;i<=n;++i) {
        int x=read(),y=read();
        int fx=find(x),fy=find(y);
        if(fx==fy) point[fx]=0;
        else {
            point[fx]=max(point[fx],point[fy]);
            fa[fy]=fx;
        }
    }
    for(int i=1;i<=10000;++i) {
        int fx=find(i);
        if(point[fx]==i) {printf("%d\n",i-1);break;}
        else if(i==10000) printf("%d\n",10000);
    }
    return 0;
}

你可能感兴趣的:(【数据结构】并查集,并查集)