zoukankan      html  css  js  c++  java
  • Codeforces Round #680 E. Team-Building(可撤销并查集)

    Codeforces Round #680 E. Team-Building(可撤销并查集)

    Codeforces Round #680 (Div. 2, based on Moscow Team Olympiad)

    题意:给n个点与m个边与k种颜色,每个点都有k中的一种颜色,要求对每对颜色,若该对颜色的点可以构成二分图则贡献为1,求总贡献量;

    题解:K的范围为5e5,枚举颜色对显然超时,但是m只有5e5,所以极限情况下也只有5e5对颜色没有贡献,所以我们可以反过来,先求出总贡献量,再减去没有贡献的数量。再讨论一下二分图,相同颜色的点可以用并查集判断是否构成二分图,同时很容易可以知道,若一种颜色本身不能构成二分图的话那么该颜色与任何颜色构成二分图,所以这部分可以特判去除。然后要在枚举两种颜色时,应先删除上一次枚举对并查集的影响,这里我们用可撤销并查集维护即可。

        #include<iostream>
        #include<algorithm>
        #include<stack>
        #include<vector>
        using namespace std;
        int n,m,k,u,v;
        int c[1000007];
        int fa[1000007];
        int sz[1000007];
        int vis[1000007];
        stack<pair<int,int>>sa;
        struct madoka{
        	int u;
        	int v;
        	int cu;
        	int cv;
        };
        bool cmp(madoka a1,madoka a2){
        	if(a1.cv==a2.cv){
        		return a1.cu<a2.cu;
        	}
        	return a1.cv<a2.cv;
        }
        vector<madoka>ho;
        int fin(int p){
        	if(p==fa[p])return p;
        	else{
        		return fin(fa[p]);
        	}
        }
        void merge(int u,int v){
        	int f1=fin(u);
        	int f2=fin(v);
        	if(sz[f1]>sz[f2]){
        		swap(f1,f2);
        	}
        	fa[f1]=f2;
        	sz[f2]+=sz[f1];
        	sa.push({f1,f2});
        }
        void clr(){
        	while(!sa.empty()){
        		int f1=sa.top().first;
        		int f2=sa.top().second;
        		sa.pop();
        		fa[f1]=f1;
        		sz[f2]-=sz[f1];
        	}
        }
        int main(){
        	scanf("%d%d%d",&n,&m,&k);
        	for(int i=1;i<=n;i++){
        		scanf("%d",&c[i]);
        	}
        	for(int i=1;i<=n*2;i++){
        		fa[i]=i;
        		sz[i]=1;
        	}
        	long long ans=k;
        	for(int i=1;i<=m;i++){
        		scanf("%d%d",&u,&v);
        		if(c[u]==c[v]){
        			int f1=fin(u);
        			int f2=fin(v);
        			if(f1==f2&&vis[c[u]]==0){
        				ans--;
        				vis[c[u]]=1;
        				continue;
        			}
        			merge(u,v+n);
        			merge(v,u+n);
        		}
        		else{
        			if(c[u]>c[v]){
        				swap(u,v);
        			}
        			ho.push_back({u,v,c[u],c[v]});
        		}
        	}
        	ans=ans*(ans-1)/2;
        	sort(ho.begin(),ho.end(),cmp);
        	while(!sa.empty()){
        		sa.pop();
        	}
        	int r;
        	for(int i=0;i<ho.size();i=r+1){
        		clr();
        		int c1=ho[i].cu;
        		int c2=ho[i].cv;
        		r=i;
        		for(int j=i;j<ho.size();j++){
        			if(ho[j].cu==c1&&ho[j].cv==c2){
        				r=j;
        			}
        			else{
        				break;
        			}
        		}
        		if(vis[c1]||vis[c2]){
        			continue;
        		}
        		int ok=0,u,v;
        		for(int j=i;j<=r;j++){
        			u=ho[j].u;
        			v=ho[j].v;
        			if(fin(u)==fin(v)){
        				ok=1;
        				break;
        			}
        			else{
        				merge(u+n,v);
        				merge(u,v+n);
        			}
        		}
        		ans-=ok;
        	}
        	printf("%lld
    ",ans);
        }
    
    
    
  • 相关阅读:
    Pytorch手写线性回归
    numpy+sklearn 手动实现逻辑回归【Python】
    如何用TensorFlow实现线性回归
    进程、线程和携程的通俗解释【刘新宇Python】
    即时通信WebSocket 和Socket.IO
    gRPC【RPC自定义http2.0协议传输】
    Django中MySQL事务的使用
    模拟电磁曲射炮_H题 方案分析【2019年电赛】【刘新宇qq522414928】
    Gitflow工作流
    雪花算法【分布式ID问题】【刘新宇】
  • 原文地址:https://www.cnblogs.com/whitelily/p/13962478.html
Copyright © 2011-2022 走看看