• ACdream1726-A Math game+(DFS+二分)+(DFS+前缀和)


    传送门

    官方题解:http://acdream.info/topic?tid=4246

    参考:https://www.cnblogs.com/nowandforever/p/4492428.html

    题意:在给定的n个数中,能否找到几个数使得这几个和等于H;

    思路:注意这道题的条件 0<n<=40, 0<=H<10^9, 0<=a[i]<10^9,其中 H 和 A [i] 给的比较大,dp的空间开不下,而n比较小,只有40,所以可以把所给的数分成20、20两部分,用dfs或者直接状态压缩从0至2^(n/2)循环来搞出每组所有可能的和(dfs+剪枝应该比循环快),然后再在第二部分找(H-第一部分的可能和)就可以了。卡了一下map和set。可以用hash或者二分来找

    ac代码(600+ms):

    #include <cstdio>
    #include <algorithm>
    using namespace std;
    
    const int maxn = 1050000;        //比2的20次大一点
    
    int n,h,a[50],d[50],c[maxn],cnt,flag;
    
    int check(int s)
    {
        int le=1,ri=cnt;
        while(le<=ri)
        {
            int mid=(le+ri)>>1;
            if(c[mid]==s)return 1;
            if(c[mid]>s)
                ri = mid - 1;
            else le    = mid + 1; 
        }
        return 0;
    }
    
    void dfs(int sum,int cur,int nn,int on)//利用on使得两次dfs放在了一起
    {
        if(flag)return;
        if(cur==nn+1)
        {
            if(on)
                c[++cnt]=sum;
            else flag = check(h-sum);
            return;
        }
        for(int i=0;i<2;i++)
        {
            int t=sum+d[cur]*i;
            if(t > h)return;
            dfs(t,cur+1,nn,on);
        }
    }
    int main(){
        //freopen("in","r",stdin);
        while(~scanf("%d%d",&n,&h))
        {
            for(int i = 1; i <= n; i++)
            {
                scanf("%d",&a[i]);
            }
            cnt = flag = 0;
            int n1 = (n>>1), n2=n-n1;//(n>>1)没加括号WA了几次
            for(int i=1; i<=n1; i++)
            {
                d[i]=a[i];
            }
            dfs(0,1,n1,1);     //第一次先算出前一半的所有可能和;
            sort(c+1,c+cnt+1);
            for(int i=n1+1;i<=n;i++)
            {
                d[i-n1]=a[i];
            }
            dfs(0,1,n2,0);    //第二次求后一半的和 ,在用二分在前一半中找有没有对应的值
            if(flag)puts("Yes");
            else puts("No");
        }
    
        return 0;
    }
    View Code

    我一开始注意到n比较小,就直接dfs暴力,但是超时了;

    不过,我又看到有人没有用二分,只用了dfs+前缀和,0ms;

    自己写的(12+ms)

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int maxn = 50;
    int h,a[maxn],sum[maxn],n,ans=0;
    void dfs(int cur,int s)
    {
        if(ans)return;
        if(s>sum[cur])return;
        if(sum[cur]==s||s==0){ans=1;return;}
       
       for(int i=n;i>=1;i--)
       {
            if(ans)return;
            if(a[i]<=s)
            {
                dfs(i-1,s-a[i]);
            }
       }
    }
    int main(){
       // freopen("in","r",stdin);
        while(~scanf("%d%d",&n,&h))
        {
            sum[0]=0;
            for(int i=1;i<=n;i++)
            {
                scanf("%d",&a[i]);
            }
            sort(a+1,a+1+n);        //不sort一直超时
            for(int i=1;i<=n;i++)
            {
                sum[i]=sum[i-1]+a[i];
            }
            ans=0;
            dfs(n,h);
            if(ans)puts("Yes");
            else puts("No");
        }
     
        return 0;
    }
    View Code
  • 相关阅读:
    协方差矩阵
    VS2010+C#+EmguCV 配置详解
    OpenCv,EmguCv及.net之间的互动(The Interaction of OpenCv, EmguCv AND .net)
    EmguCV学习 与opencv的区别和联系
    redis新手入门,摸不着头脑可以看看<二>
    java常用工具类[待补充]
    redis新手入门,摸不着头脑可以看看<一>
    用java代码发送http请求
    Date和long类型互转
    WEB-INF目录下文件复制的几种方式
  • 原文地址:https://www.cnblogs.com/ckxkexing/p/8562433.html
Copyright © 2020-2023  润新知