码迷,mamicode.com
首页 > 其他好文 > 详细

[AGC012D]Colorful Balls

时间:2017-10-26 23:20:26      阅读:251      评论:0      收藏:0      [点我收藏+]

标签:void   nbsp   include   highlight   amp   text   ati   oid   stdio.h   

题意:有$N$个球,有颜色$c_i$,重量$w_i$,若($c_a=c_b$且$w_a+w_b\leq X$)或($c_a\ne c_b$且$w_a+w_b\leq Y$),可以交换$a,b$,求总共可以得到多少种不同的颜色序列

 

首先我们当然要比较快地找到哪些球是可以交换的

因为交换具有传递性,所以我们构造一个图:若$i$和$j$可以交换,他们之间有至少一条路径

先看颜色相同

设$m_i$表示颜色为$i$的球中重量最小的球

那么对于每一个球$k$,只需要考虑$w_k+w_{m_{c_k}}\leq X$即可,因为所有球都与最轻的相连,他们也就自然连通了

再看颜色不同

设$m_i$中重量最小的球为$a$,重量次小的球为$b$

那么对于每一个球$k$,只需要考虑$w_k+w_a\leq Y$或$w_k+w_b\leq Y$即可

证明:对于任意$s,t(w_s+w_t\leq Y)$

若$c_s\ne c_a$且$c_t\ne c_a$,交换方式:$s\Leftrightarrow a\Leftrightarrow t$

若$c_s\ne c_b$且$c_t\ne c_b$,交换方式:$s\Leftrightarrow b\Leftrightarrow t$

若$\{c_s,c_t\}=\{c_a,c_b\}$,交换方式:$s\Leftrightarrow a\Leftrightarrow b\Leftrightarrow t$

于是我们得到哪些球可以随意交换,现在要算方案数

若有$m$种颜色,第$i$种颜色的数量为$v_i$

我们按顺序把每种颜色gay进去

轮到第$i$种颜色时,有$v_i+\cdots+v_m$个空位,方案数为$C_{v_i}^{v_i+\cdots+v_m}$

于是总方案数为$C_{v_1}^{v_1+\cdots+v_m}\cdot C_{v_2}^{v_2+\cdots+v_m}\cdot\cdots\cdot C_{v_m}^{v_m}$

都是$n$的级别,直接预处理阶乘暴力算即可

#include<stdio.h>
#define mod 1000000007ll
#define inf 2147483647ll
#define ll long long
struct edge{
	int to,nex;
}e[3000010];
int c[200010],h[200010],pos[200010],cnt[200010],tot;
ll w[200010],fac[200010],rfac[200010],min[200010];
bool v[200010];
ll pow(ll a,ll b){
	ll ans=1,base=a;
	while(b){
		if(b&1)ans=(ans*base)%mod;
		base=(base*base)%mod;
		b>>=1;
	}
	return ans;
}
ll C(ll n,ll k){
	return(((fac[n]*rfac[n-k])%mod)*rfac[k])%mod;
}
void add(int a,int b){
	tot++;
	e[tot].to=b;
	e[tot].nex=h[a];
	h[a]=tot;
}
void dfs(int x){
	v[x]=1;
	for(int i=h[x];i;i=e[i].nex){
		if(!v[e[i].to])dfs(e[i].to);
	}
}
int main(){
	int n,i,fir,sec=0;
	ll X,Y,min1,min2,tot,ans;
	scanf("%d%lld%lld",&n,&X,&Y);
	fac[0]=1;
	for(i=1;i<=n;i++){
		scanf("%d%lld",c+i,w+i);
		fac[i]=(i*fac[i-1])%mod;
		min[i]=inf;
	}
	rfac[n]=pow(fac[n],mod-2);
	for(i=n;i>0;i--)rfac[i-1]=(i*rfac[i])%mod;
	for(i=1;i<=n;i++){
		if(w[i]<min[c[i]]){
			min[c[i]]=w[i];
			pos[c[i]]=i;
		}
	}
	for(i=1;i<=n;i++){
		if(i!=pos[c[i]]&&w[i]+min[c[i]]<=X){
			add(i,pos[c[i]]);
			add(pos[c[i]],i);
		}
	}
	min1=min2=inf;
	for(i=1;i<=n;i++){
		if(w[i]<min1){
			min1=w[i];
			fir=i;
		}
	}
	for(i=1;i<=n;i++){
		if(w[i]<=min2&&c[i]!=c[fir]){
			min2=w[i];
			sec=i;
		}
	}
	for(i=1;i<=n;i++){
		if(i==fir)continue;
		if(c[i]!=c[fir]&&w[i]+w[fir]<=Y){
			add(i,fir);
			add(fir,i);
		}
		if(sec&&c[i]!=c[sec]&&w[i]+w[sec]<=Y){
			add(i,sec);
			add(sec,i);
		}
	}
	dfs(fir);
	tot=0;
	for(i=1;i<=n;i++){
		if(v[i]){
			tot++;
			cnt[c[i]]++;
		}
	}
	ans=1;
	for(i=1;i<=n;i++){
		if(cnt[i]){
			ans=(ans*C(tot,cnt[i]))%mod;
			tot-=cnt[i];
		}
	}
	printf("%lld",ans);
}

[AGC012D]Colorful Balls

标签:void   nbsp   include   highlight   amp   text   ati   oid   stdio.h   

原文地址:http://www.cnblogs.com/jefflyy/p/7739262.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!