3139: [Hnoi2013]比赛

Description

沫沫非常喜欢看足球赛,但因为沉迷于射箭游戏,错过了最近的一次足球联赛。此次联 赛共N支球队参加,比赛规则如下: 
(1) 每两支球队之间踢一场比赛。 (2) 若平局,两支球队各得1分。 
(3) 否则胜利的球队得3分,败者不得分。 
尽管非常遗憾没有观赏到精彩的比赛,但沫沫通过新闻知道了每只球队的最后总得分, 然后聪明的她想计算出有多少种可能的比赛过程。 
譬如有3支球队,每支球队最后均积3分,那么有两种可能的情况:
 可能性1    可能性2 
球队  A  B  C  得分   球队 A  B  C  得分 
A        -  3  0  3             A     -  0  3  3 
B        0  -  3  3             B    3  -  0  3 
C        3  0  -  3            C    0  3  -  3 
但沫沫发现当球队较多时,计算工作量将非常大,所以这个任务就交给你了。请你计算 出可能的比赛过程的数目,由于答案可能很大,你只需要输出答案对109+7取模的结果

题解:

记忆化搜索,既然要记忆化,我们肯定是保存得分状态啦,用hash来搞搞。

因为先搞哪个队对结果没有影响,所以我们先排个序,再来一层搜索枚举出当前队和剩余队的输赢情况,转化成子问题继续递归处理。

还有个剪枝,比较水,可以直接去看看代码.

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<map>
#define ll long long
#define mod 1000000007
using namespace std;
int n;
ll a[20],b[20];
map<ll,ll>mp;
bool cmp(ll x,ll y)
{
	return x>y;
}
ll hash(int x)
{
	ll tmp=x;
	for(int i=1;i<=x;i++) b[i]=a[i];
	sort(b+1,b+1+x,cmp);
	for(int i=1;i<=x;i++) tmp+=tmp*28+b[i];
	return tmp;
}

ll dfs(int x,int y)
{
	if(a[y]>3*(y-x)) return -1;//剪枝 
	ll ans=0;
	if(x==y)
	{
		if(y==1) return 1;
		else
		{
			ll yu=hash(y-1);//得到当前状态 
			if(mp[yu]) return mp[yu];
			return mp[yu]=dfs(1,y-1);//当前状态有多少种可能 
		}
	}
	if(a[y]>=3)
	{
		a[y]-=3;
		ll tmp=dfs(x+1,y);
		if(tmp!=-1)ans=(ans+tmp)%mod;
		a[y]+=3;
	}
	if(a[y]&&a[x])
	{
		a[y]--;a[x]--;
		ll tmp=dfs(x+1,y);
		if(tmp!=-1) ans=(ans+tmp)%mod;
		a[y]++;a[x]++;
	}
	if(a[x]>=3)
	{
		a[x]-=3;
		ll tmp=dfs(x+1,y);
		if(tmp!=-1) ans=(ans+tmp)%mod;
		a[x]+=3;
	}
	if(ans!=0) return ans;
	else return -1;
}
int main()
{
	scanf("%d",&n);
	for(int i=1;i<=n;i++)
	{
		scanf("%lld",&a[i]);
	}
	sort(a+1,a+1+n,cmp);
	printf("%lld",dfs(1,n));
}

好像整个竞赛室都感冒了(滑稽),整个下午头都晕晕的。

超想睡觉QAQ。。。。。。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值