zoukankan      html  css  js  c++  java
  • [洛谷P1196]银河英雄传说

    题目大意:有30000个船,第i个船在第i列,两个操作,M:把第i列的船整体拼到第j列船后,C:求第i和第j个船之间的船的个数
    题解:可以想到用并查集,在基础的并查集上增加路径,用num数组存第i个船到队首的距离,用len存这一列的船的个数。问题主要在路径压缩上维护num数组上。可以在find的过程中,每改变f[x]的时候把num[i]加上num[f[i]],就可以维护了。
    卡点:在num数组初始赋值为了1,导致每find一遍,所搜索的值就加了1(根节点)。没看清要求求的是中间船的个数,写成了包括两头的。在求出答案时,误把num[i]-num[j]写成了num[find[i]]-num[find[j]]

    C++ Code:

    #include<cstdio>
    #include<cstdio>
    const int maxn=30010;
    int n,f[maxn],num[maxn],len[maxn];
    char ch[5];
    int find(int x){
        if (x==f[x])return x;
        int xx=find(f[x]);
        num[x]+=num[f[x]];
        return f[x]=xx;
    }
    int abs(int a){return a>0?a:-a;}
    int main(){
        scanf("%d
    ",&n);
        for (int i=1;i<maxn;i++)f[i]=i,len[i]=1;
        while (n--){
            int a,b;
            scanf("%s%d%d",ch,&a,&b);
            if (ch[0]=='M'){
                int x=find(a),y=find(b);
                if (x==y)continue;
                num[x]=len[y];
                len[y]+=len[x];
                f[x]=y;
            }else{
                int x=find(a),y=find(b);
                if (x!=y){
                    puts("-1");
                    continue;
                }
                printf("%d
    ",abs(num[a]-num[b])-1);
            }
        }
        return 0;
    }
    
  • 相关阅读:
    巩固复习(对以前的随笔总结)_01
    Django 项目分析后得到的某些结论
    django 命令行命令
    实现搜索视频到播放(非原创)
    python 打包
    随笔汇总,温故知新
    找伙伴
    sam-Toy Cars
    反质数
    Blue Mary的战役地图
  • 原文地址:https://www.cnblogs.com/Memory-of-winter/p/8017137.html
Copyright © 2011-2022 走看看