zoukankan      html  css  js  c++  java
  • bzoj1483: [HNOI2009]梦幻布丁

     传送门

    每次只要修改链头实际代表的颜色即可

    //Twenty
    #include<cstdio>
    #include<cstdlib>
    #include<iostream>
    #include<algorithm>
    #include<cmath>
    #include<cstring>
    #include<queue>
    #include<vector>
    using namespace std;
    const int maxn=1000000+299;
    int n,m,xx,yy,opt,ans,v[maxn],a[maxn],fir[maxn],nxt[maxn],sz[maxn];
    void add(int xx,int yy){
        nxt[yy]=fir[xx]; fir[xx]=yy; sz[xx]++;
    }
    void change(int xx,int yy){
        if(!fir[xx]||xx==yy) return;
        for(int i=fir[xx];i;i=nxt[i]){
            if(i>1&&a[i-1]==yy) ans--;
            if(i<n&&a[i+1]==yy) ans--;
        }
        for(int i=fir[xx];i;i=nxt[i]){
            a[i]=yy;
            if(!nxt[i]) {nxt[i]=fir[yy];break;}
        }
        sz[yy]+=sz[xx]; fir[yy]=fir[xx]; 
        sz[xx]=0; fir[xx]=0;
    
    }
    int main()
    {
        scanf("%d%d",&n,&m);
        for(int i=1;i<=n;i++){
            scanf("%d",&a[i]);
            if(i==1||a[i]!=a[i-1]) ans++;
            add(a[i],i);
            
        }
        for(int i=1;i<=maxn;i++) v[i]=i;//可能有一开始不存在的颜色出现 
        while(m--){
            scanf("%d",&opt);
            if(opt==1){
                scanf("%d%d",&xx,&yy);
                if(sz[v[xx]]>sz[v[yy]]) swap(v[xx],v[yy]);
                //改的时候改代表这个颜色的现在是哪个颜色的链表就好 
                change(v[xx],v[yy]);
            }
            else printf("%d
    ",ans);
        }
        return 0;
    }
    梦幻布丁

     

  • 相关阅读:
    数组中寻找和为X的两个元素
    JSP&Servlet学习笔记(一)
    自下而上的动态规划算法
    计数排序
    快速排序
    堆排序
    LeetCode-001题解
    算法不归路之最大子序列(C++版)
    算法不归路之插入排序(C版)
    互联网五层模型
  • 原文地址:https://www.cnblogs.com/Achenchen/p/7474966.html
Copyright © 2011-2022 走看看