博客园 首页 私信博主 显示目录 隐藏目录 管理 动画

BZOJ.3139.[HNOI2013]比赛(搜索 Hash)

题目链接

不会搜索了。。

DFS()中两个参数,枚举每两个队伍的比赛结果(分配当前队伍的分数)。
可以发现方案数量与具体哪只球队得了多少分无关,只与当前比赛的队伍数量和得分序列的组成有关。可以记忆化搜索。
DFS()中是从某支队伍和它后面的队伍一一进行比赛 分配得分,分配完当前后将其它队伍的得分情况哈希,看是否算过;
如果没就进行下一支队伍,其实就是计算去掉当前这支队伍的方案数。
n(n<=10)支队伍分数的情况最多为(289?)1014级别,可以直接用longlong存。

这样做为什么必须要降序排?
因为前面小的\(val\)更可能变得\(<3\),从而省去后面的DFS。

//1616kb	684ms
#include <map>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mod (1000000007)
typedef long long LL;
const int N=11;

int n,val[N];
std::map<LL,int> sta[N];//应该是每一位开一个map?
std::map<LL,int>::iterator it;

inline bool cmp(int a,int b){
	return a<b;
}
LL Get_Hash(int p)
{
	int tmp[N];
	for(int i=p; i<=n; ++i) tmp[i]=val[i];
	std::sort(tmp+p,tmp+n+1,cmp);//tmp+p!不是tmp+p+1 
	LL res=0;
	for(int i=p; i<=n; ++i) res=res*28+tmp[i];//tmp[i] not val[i]!(这还能写错 我说怎么还T)
	return res;
}
int DFS(int l,int r)
{
	if(val[l]>(r-l)*3) return 0;//根据上面的剪枝,不可能合法!
	if(l==r)
	{
//		if(val[l]) return 0;//只有把分数恰好分配完才合法!
		if(l==n) return 1;
		LL s=Get_Hash(l+1);
		if((it=sta[l].find(s))!=sta[l].end()) return it->second;//
		return sta[l][s]=DFS(l+1,n);//这不能给迭代器(sta[l].end())赋值啊
	}
	int res=0;
	if(val[l]>=3)
	{
		val[l]-=3;
		res+=DFS(l,r-1), res>=mod?res-=mod:0;
		val[l]+=3;
	}
	if(val[l]&&val[r])
	{
		--val[l], --val[r];
		res+=DFS(l,r-1), res>=mod?res-=mod:0;
		++val[l], ++val[r];
	}
	if(val[r]>=3)
	{
		val[r]-=3;
		res+=DFS(l,r-1), res>=mod?res-=mod:0;
		val[r]+=3;
	}
	return res;
}

int main()
{
	scanf("%d",&n);
	for(int i=1; i<=n; ++i) scanf("%d",&val[i]);
	std::sort(val+1,val+1+n,cmp);
	printf("%d",DFS(1,n));

	return 0;
}
posted @ 2018-03-29 16:09  SovietPower  阅读(205)  评论(0编辑  收藏  举报