[HNOI2002]彩票

Posted peng-ym

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[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;
}

以上是关于[HNOI2002]彩票的主要内容,如果未能解决你的问题,请参考以下文章

bzoj2002 [Hnoi2010]Bounce 弹飞绵羊LCT

bzoj2002: [Hnoi2010]Bounce 弹飞绵羊

bzoj1588: [HNOI2002]营业额统计 splay瞎写

BZOJ1588: [HNOI2002]营业额统计

bzoj 2002 [Hnoi2010]Bounce 弹飞绵羊 LCT

luogu P2234 [HNOI2002]营业额统计