zoukankan      html  css  js  c++  java
  • 【DFS】NYOJ-325-zb的生日

    【题目链接:NYOJ-325】

      一道以我名字命名的题目,难道要我生日的时候再A?

      思路:依旧深搜,但这个问题应该有一个专有名词吧,看别的博客说是 “容量为 sum/2 的背包问题”,不懂。。。

     1 // abs()  对应头文件 stdlib.h 返回int参数 
     2 // fabs() 对应头文件 math.h   返回double参数 
     3 #include<cstdio>
     4 #include<cstring>
     5 #include<stdlib.h> 
     6 int W[22],sum,m,n;
     7 void dfs(int cur,int total){
     8     int t =abs(sum - total * 2);
     9     if(t < m) 
    10         m = t;
    11     if(cur == n){
    12 //        int t =abs(sum - total * 2);
    13 //        if(t < m)
    14 //            m = t;
    15         return;//递归边界 
    16     } 
    17     if(total > sum / 2){    //剪枝 
    18 //        int t =abs(sum - total * 2);
    19 //        if(t < m)
    20 //            m = t;
    21         return;
    22     }
    23     dfs(cur + 1,total + W[cur]);    //结点分支。加,右分支 
    24     dfs(cur + 1,total);             //节点分支。不加,左分支 
    25 } 
    26 int main(){    
    27     while(~scanf("%d",&n)){
    28         memset(W,0,sizeof(W));
    29         sum = 0;
    30         m = 10001;
    31         for(int i = 0;i < n;i++){
    32             scanf("%d",&W[i]);
    33             sum += W[i];
    34         }
    35         dfs(0,0);
    36         printf("%d
    ",m);
    37     }
    38     return 0; 
    39 }

      渐渐对搜索有了一些认识,刚开始完全无头绪,要多练习,加油!

      最优解:

        也就是动脑筋剪枝。

     1 #include<cstdio>
     2 #include<cstring>
     3 #include<algorithm>
     4 using namespace std;
     5 
     6 int a[25];
     7 int s[25];
     8 int n;
     9 int V;
    10 int ans;
    11 
    12 void DFS(int i,int v)
    13 {
    14     if(i==n+1)
    15     {
    16         ans=max(ans,v);
    17         return;
    18     }
    19     //如果背包已经装满,则不再考虑其他情况。
    20     //如果背包中已有物品加上现有可选物品的总重量都不大于已知的最优解,则剪枝
    21     if(ans==V||v+s[n]-s[i-1]<=ans)
    22         return;
    23     if(a[i]+v<=V)
    24     {
    25         DFS(i+1,v+a[i]);
    26     }
    27     DFS(i+1,v);
    28 }
    29 
    30 int main()
    31 {
    32     while(scanf("%d",&n)==1)
    33     {
    34         s[0]=0;
    35         for(int i=1;i<=n;i++)
    36         {
    37             scanf("%d",&a[i]);
    38             s[i]=s[i-1]+a[i];
    39         }
    40         V=s[n]/2;
    41         ans=0;
    42         DFS(1,0);
    43         ans=s[n]-ans-ans;
    44         printf("%d
    ",ans);
    45     }
    46     return 0;
    47 }

      

  • 相关阅读:
    19牛客暑期多校 round2 H 01矩阵内第二大矩形
    NOIP2017滚粗记
    Left 4 Dead 2(求生之路2) 游戏打不开 游戏闪退 的一种可能性以及解决方法
    Luogu P1156 垃圾陷阱
    Luogu P1376 机器工厂
    Luogu P1842 奶牛玩杂技
    Luogu P1880 石子合并
    Luogu P1441 砝码称重(fj省选)
    Luogu P1077 摆花
    Luogu P1282 多米诺骨牌
  • 原文地址:https://www.cnblogs.com/zhengbin/p/4486572.html
Copyright © 2011-2022 走看看