CF878CTournament set+并查集+链表

Posted CQzhangyu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CF878CTournament set+并查集+链表相关的知识,希望对你有一定的参考价值。

【CF878C】Tournament

题意:有k个项目,n个运动员,第i个运动员的第j个项目的能力值为aij。一场比赛可以通过如下方式进行:

每次选出2个人和一个项目,该项目能力值高者获胜,败者被淘汰,胜者继续比赛。最后一个人是冠军。

在一场比赛中,你可以任意安排比赛顺序,任意选择每次的参赛者和项目,现在想知道的是有多少人可能成为最后的冠军。

为了加大难度,一共有n次询问,第i次询问是前i个人进行比赛,问最终由多少人可能成为总冠军。

n<=50000,k<=10,aij<=10^9

题解:只要敢想就去写吧。

我们将所有人看成一张n个点的有向图,如果i的某项能力值比j高,则从i到j连一条有向边。我们将得到的整个图中的强连通分量缩成点,那么最终得到的一定是一条链。其中每个强联通分量中每一项的最小值都比下一个强联通分量的最大值还大。然后我们依次加入每个点,考虑如何维护这条链。

在加入第i个人时,对于每个项目,我们可以在set中找到这个人的前驱和后继,并记录二者在链中的位置。令a为每个项目中前驱位置的最大值,b为每个项目中后继位置的最小值,如果a>b,则说明i能打过a,b能打过i,并且a能打过b,出现了一个环!我们将这个环暴力缩掉即可;如果a=b,我们将i加入到a的强联通分量即可;如果a<b,那么a=b-1,我们把i加到ab中间即可。

可以用并查集维护连通性,链表维护整条链。

 

#include <cstdio>
#include <cstring>
#include <iostream>
#include <set>
#include <utility>
#define mp(A,B) make_pair(A,B)
using namespace std;
const int maxn=50010;
int n,m,last;
int f[maxn],mx[maxn][11],mn[maxn][11],siz[maxn],nxt[maxn],v[11];

set<pair<int,int> > s[11];
set<pair<int,int> >::iterator it;

inline int rd()
{
	int ret=0,f=1;	char gc=getchar();
	while(gc<‘0‘||gc>‘9‘)	{if(gc==‘-‘)	f=-f;	gc=getchar();}
	while(gc>=‘0‘&&gc<=‘9‘)	ret=ret*10+(gc^‘0‘),gc=getchar();
	return ret*f;
}
int find(int x)
{
	return (f[x]==x)?x:(f[x]=find(f[x]));
}
inline void merge(int a,int b)
{
	for(int i=1;i<=m;i++)	mx[b][i]=max(mx[b][i],mx[a][i]),mn[b][i]=min(mn[b][i],mn[a][i]);
	siz[b]+=siz[a],f[a]=b;
}
int main()
{
	n=rd(),m=rd();
	int i,j,a,b,t;
	for(j=1;j<=m;j++)	s[j].insert(mp(mx[1][j]=mn[1][j]=rd(),1));
	f[1]=siz[1]=last=1;
	puts("1");
	for(i=2;i<=n;i++)
	{
		a=b=0;
		for(j=1;j<=m;j++)
		{
			v[j]=rd();
			it=s[j].upper_bound(mp(v[j],i));
			if(it!=s[j].end())
			{
				t=find((*it).second);
				if(!b||mx[t][j]<mx[b][j])	b=t;
			}
			if(it!=s[j].begin())
			{
				it--,t=find((*it).second);
				if(!a||mx[t][j]>mx[a][j])	a=t;
			}
			s[j].insert(mp(v[j],i));
		}
		f[i]=i,siz[i]=1;
		for(j=1;j<=m;j++)	mx[i][j]=mn[i][j]=v[j];
		if(!a)	nxt[i]=b;
		else	if(!b)	last=i,nxt[a]=i;
		else	if(a==b)	merge(i,a);
		else	if(mx[a][1]<mx[b][1])	nxt[a]=i,nxt[i]=b;
		else
		{
			for(t=b;t!=a;t=nxt[t],merge(t,b));
			merge(i,b);
			nxt[b]=nxt[a];
			if(a==last)	last=b;
		}
		printf("%d\n",siz[last]);
	}
	return 0;
}

 

以上是关于CF878CTournament set+并查集+链表的主要内容,如果未能解决你的问题,请参考以下文章

CF553C Love Triangles(带权并查集)

cf 并查集

并查集+multiset+双指针——cf982D

CF745 C 并查集

并查集CF731CSocks

cf246 ENew Reform (并查集找环)