[HNOI2002]彩票

  • 先说一下题意:很显然,题意要求从m个自然数中选择n个使得倒数之和为x/y,输出满足条件的方案数。(m<=50,n<=10)
  • 数据范围就一定说明了很多,显然就是一个搜索剪枝。最基本的剪枝就是如果比x/y大就返回。还有两个剪枝就比较常见了。1.如果当前的值加上最大可能的值小于答案那就返回;2.如果当前的值加上最小可能的值大于答案那就返回。我抱着试一试的想法,然后就A了。不过肯定还有更多的剪枝,但是这样已经能过。
  • 当然,要注意精度误差
  • 下附代码:
#include<bits/stdc++.h>
using namespace std;
inline void read(int &x)
{
	x=0;
	static int p;p=1;
	static char c;c=getchar();
	while(!isdigit(c)){if(c=='-')p=-1;c=getchar();}
	while(isdigit(c)) {x=(x<<1)+(x<<3)+(c-48);c=getchar();}
	x*=p;
}
const double eps=1e-10;
int n,m,x,y,ans;
double tag;
bool vis[60];
void dfs(int x,double sum,int last)
{
	if(sum-tag>eps)return;
	if(sum+(double)(n-x+1)*1.0/(double)(last+1)+eps<tag)return;
	if((sum+(double)(n-x+1)*1.0/(double)m)>tag+eps)return;
	if(x==n+1)
	{
		if(fabs(sum-tag)<=eps)ans++;
		return ;
	}
	for(int i=last+1;i<=m;i++)
	{
		if(!vis[i])
		{
			vis[i]=true;
			dfs(x+1,sum+1.0/(double)i,i);
			vis[i]=false;
		}
	}
}
int main()
{
//	freopen(".in","r",stdin);
//	freopen(".out","w",stdout);
	read(n);read(m);read(x);read(y);
	tag=(double)x/(double)y;
	dfs(1,0,0);
	cout<<ans<<endl;
	return 0;
}
posted @ 2018-03-25 22:27  pengym  阅读(226)  评论(0编辑  收藏  举报