Codeforces 840C. On the Bench 动态规划 排列组合

原文链接https://www.cnblogs.com/zhouzhendong/p/CF840C.html

题解

首先,我们可以发现,如果把每一个数的平方因子都除掉,那么剩下的数,不相等的数都可以相邻,相等的数都不能相邻。

也就是说我们把所有数分成了一些集合,同一个集合内的元素不能相邻,不同集合之间的元素可以相邻。

关键部分到了!

设 $dp[i][j]$ 表示前 $i$ 个集合,有 $j$ 对相邻元素相同的方案数。

转移的时候枚举一下把当前集合分成多少段,有多少段插在之前的相同相邻元素之间。

由于所有集合的size 加起来是 n ,所以时间复杂度不是 $O(n^4)$,是 $O(n^3)$ 的。

代码

#pragma GCC optimize(2)
#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
#define For(i,a,b) for (int i=a;i<=b;i++)
#define Fod(i,b,a) for (int i=b;i>=a;i--)
using namespace std;
typedef long long LL;
LL read(){
	LL x=0,f=0;
	char ch=getchar();
	while (!isdigit(ch))
		f|=ch=='-',ch=getchar();
	while (isdigit(ch))
		x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
	return f?-x:x;
}
const int N=305,mod=1e9+7;
int n;
int a[N],vis[N],s[N];
int v[N],vc=0;
int dp[N][N];
bool check(int x,int y){
	int g=__gcd(x,y);
	x/=g,y/=g;
	int sqx=sqrt(x),sqy=sqrt(y);
	return sqx*sqx==x&&sqy*sqy==y;
}
void Add(int &x,int y){
	if ((x+=y)>=mod)
		x-=mod;
}
int C[N][N],Fac[N];
int main(){
	n=read();
	For(i,1,n)
		a[i]=read();
	clr(vis);
	For(i,1,n)
		if (!vis[i]){
			int cnt=0;
			For(j,i,n)
				if (!vis[j]&&check(a[i],a[j]))
					vis[j]=1,cnt++;
			v[++vc]=cnt;
			s[vc]=s[vc-1]+v[vc];
		}
	for (int i=Fac[0]=1;i<N;i++)
		Fac[i]=(LL)Fac[i-1]*i%mod;
	for (int i=0;i<N;i++)
		C[i][i]=C[i][0]=1;
	for (int i=1;i<N;i++)
		for (int j=1;j<i;j++)
			C[i][j]=(C[i-1][j]+C[i-1][j-1])%mod;
	clr(dp);
	dp[0][0]=1;
	For(i,1,vc)
		For(j,0,s[i-1]+1)
			if (dp[i-1][j])
				For(k,1,min(v[i],s[i-1]+1))
					For(t,0,min(j,k))
						Add(dp[i][j+(v[i]-k)-t],(LL)dp[i-1][j]*Fac[v[i]]%mod*C[v[i]-1][k-1]%mod*C[j][t]%mod*C[s[i-1]+1-j][k-t]%mod);
	cout<<dp[vc][0]<<endl;
	return 0;
}

  

posted @ 2019-03-02 11:21  zzd233  阅读(291)  评论(0编辑  收藏  举报