1082: [SCOI2005]栅栏

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 2340  Solved: 991
[Submit][Status][Discuss]

Description

  农夫约翰打算建立一个栅栏将他的牧场给围起来,因此他需要一些特定规格的木材。于是农夫约翰到木材店购
买木材。可是木材店老板说他这里只剩下少部分大规格的木板了。不过约翰可以购买这些木板,然后切割成他所需
要的规格。而且约翰有一把神奇的锯子,用它来锯木板,不会产生任何损失,也就是说长度为10的木板可以切成长
度为8和2的两个木板。你的任务:给你约翰所需要的木板的规格,还有木材店老板能够给出的木材的规格,求约翰
最多能够得到多少他所需要的木板。

Input

  第一行为整数m(m<= 50)表示木材店老板可以提供多少块木材给约翰。紧跟着m行为老板提供的每一块木板的长
度。接下来一行(即第m+2行)为整数n(n <= 1000),表示约翰需要多少木材。接下来n行表示他所需要的每一块木板
的长度。木材的规格小于32767。(对于店老板提供的和约翰需要的每块木板,你只能使用一次)。

Output

  只有一行,为约翰最多能够得到的符合条件的木板的个数。

Sample Input

4
30
40
50
25
10
15
16
17
18
19
20
21
25
24
30

Sample Output

7

HINT

25切出 21 30切出 20 40切出 19、18 50切出 15、16、17

  这道题正解是二分答案+爆搜,我们可以明确一些事情,如果存在解为x那么我满足的木板的解一定有一个是从小向大排列前x个,那么我们可以对它进行二分答案。那么check就需要我们DFS了,为了剪枝,我们将所需要的木板从大向小枚举而将老板的木板从小向大枚举,如果一个老板卖的木板被我们用的小于最小的需要的木板,就将它现在的长度加入waste,代表我们用剩下的无法被利用的木料的长度,如果tot(所有老板卖的木料的总长度)-waste<sum[mid](需要的木板的前缀和)我们就直接return就好了。

  最后还有一个剪枝,如果前后两块挨着的木板长度一致,那么我们下一次搜索的起始点就一定不需要比这个点靠前,因为他们是等价的。

  这道题给了我们一些启示:

    1.搜索不一定都是那么的裸,我们还应通过他答案的性质与其他算法有机结合一下。

    2.对于一些等价搜索,我们可以利用之前的成果进行剪枝,可能会有意想不到的结果。

  1. #include<iostream>
  2. #include<cstdlib>
  3. #include<cstdio>
  4. #include<cstring>
  5. #include<queue>
  6. #include<algorithm>
  7. #include<cmath>
  8. #include<map>
  9. #define N 1055
  10. using namespace std;
  11. int n,m,c[N];
  12. int b[N],a[N],sum[N],tot;
  13. bool dfs(int k,int waste,int wz,int mid)
  14. {
  15. if(k==)return ;
  16. if(sum[mid]>tot-waste)return ;
  17. int t=waste;
  18. for(int i=wz;i<=n;i++)
  19. {
  20. if(c[i]>=a[k])
  21. {
  22. t=waste;
  23. c[i]-=a[k];
  24. if(c[i]<a[])
  25. t+=c[i];
  26. if(a[k-]==a[k])
  27. {
  28. if(dfs(k-,t,i,mid))
  29. return ;
  30. }
  31. else if(dfs(k-,t,,mid))return ;
  32. c[i]+=a[k];
  33. }
  34. }
  35. return ;
  36. }
  37. int main()
  38. {
  39. scanf("%d",&n);
  40. for(int i=;i<=n;i++)
  41. {
  42. scanf("%d",&b[i]);
  43. tot+=b[i];
  44. }
  45. scanf("%d",&m);
  46. for(int i=;i<=m;i++)
  47. scanf("%d",&a[i]);
  48. sort(b+,b+n+);
  49. sort(a+,a+m+);
  50. while(a[m]>b[n]) m--;
  51. for(int i=;i<=m;i++)
  52. sum[i]+=sum[i-]+a[i];
  53. int li=,ri=m;
  54. int ans;
  55. while(li<=ri)
  56. {
  57. memcpy(c,b,sizeof(b));
  58. int mid=(li+ri)/;
  59. if(dfs(mid,,,mid))li=mid+,ans=mid;
  60. else ri=mid-;
  61. }
  62. printf("%d\n",ans);
  63. return ;
  64. }

bzoj 1082: [SCOI2005]栅栏 题解的更多相关文章

  1. [BZOJ 1082] [SCOI2005] 栅栏 【二分 + DFS验证(有效剪枝)】

    题目链接:BZOJ - 1082 题目分析 二分 + DFS验证. 二分到一个 mid ,验证能否选 mid 个根木棍,显然要选最小的 mid 根. 使用 DFS 验证,因为贪心地想一下,要尽量先用提 ...

  2. bzoj 1082: [SCOI2005]栅栏

    Description 农夫约翰打算建立一个栅栏将他的牧场给围起来,因此他需要一些特定规格的木材.于是农夫约翰到木材店购 买木材.可是木材店老板说他这里只剩下少部分大规格的木板了.不过约翰可以购买这些 ...

  3. bzoj 1082: [SCOI2005]栅栏【二分+dfs】

    二分答案,dfs判断是否可行,当b[k]==b[k-1]时可以剪枝也就是后移枚举位置 #include<iostream> #include<cstdio> #include& ...

  4. 【BZOJ】1082: [SCOI2005]栅栏(二分+dfs)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1082 题意:n个给出木板,m个给出木板.可以将那m个木板锯成泥想要的长度.问最大能锯成多少个给出的n ...

  5. BZOJ1082: [SCOI2005]栅栏 题解

    题目大意: 有一些木材,可以没有浪费地将一根木材分成几块木板(比如长度为10的木板可以切成长度为8和2的两块木板).现在你希望得到一些长度的木板,问通过分割木材最多能得到几块想要的木板. 思路: 首先 ...

  6. 1082: [SCOI2005]栅栏

    链接 思路 二分+搜索+剪枝. 首先二分一个答案,表示最多可以切出x块.(一个结论:切出的一定是从较小的前x块.如果一个木材可以满足很多个需要的木材,那么切出最小的,就意味着以后再选时的机会更多.) ...

  7. bzoj1082: [SCOI2005]栅栏(二分答案搜索判断)

    1082: [SCOI2005]栅栏 题目:传送门 题解: 是不是一开始在想DP?本蒟蒻也是qwq,结果很nice的错了ORZ 正解:二分+搜索 我们可以先把两种木材都进行排序,那么如果需要的最大木材 ...

  8. [BZOJ1082][SCOI2005]栅栏 二分+搜索减枝

    1082: [SCOI2005]栅栏 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2430  Solved: 1034[Submit][Status ...

  9. 【BZOJ1082】[SCOI2005]栅栏(搜索)

    [BZOJ1082][SCOI2005]栅栏(搜索) 题面 BZOJ 洛谷 题解 随便写个爆搜,洛谷上就\(80\)分了.先放爆搜代码: #include<iostream> #inclu ...

随机推荐

  1. 16.Nov Working Note

    05 今天也很忙,版本发布在即,但之前的日志系统发现了bug:在中文模式下python读写抛出异常,通过转化为utf8除去异常,上传到服务器还有乱码. 另外,就是多组件安装时,多线程发生冲突.因为每一 ...

  2. java模拟post请求发送json数据

    import com.alibaba.fastjson.JSONObject; import org.apache.http.client.methods.CloseableHttpResponse; ...

  3. MongoDB centos安装问题 error while loading shared libraries: libnetsnmpmibs.so.31

    安装mongodb-linux-x86_64-enterprise-rhel70-4.0.5 cd  /usr/mongodb tar -zxvf mongodb-linux-x86_64-enter ...

  4. springboot 2.x处理404、500等异常

    404错误 404错误是不经过Controller的,所以使用@ControllerAdvice或@RestControllerAdvice无法获取到404错误 springboot2处理404错误的 ...

  5. Building PySide on Microsoft Windows

    Prerequisites MS Visual Studio Express 2008 [microsoft.com] NOTE: Visual Studio Express 2010 is not ...

  6. qt截获html请求(继承QNetworkAccessManager和QNetworkReply)

    QtWebkit加载html页面,html中会有很多的请求,比如<img id="testImg" src="http://*.jpg" width=&q ...

  7. linux dll hell--链接库real name, soname, link name

    DLL hell 是指 Windows 系统上动态库的新版本覆盖旧版本,且新版本不能兼容旧版本的问题. 例如:装新软件,但原有的软件运行不起来了.   Linux 系统下也同样面临着和 Windows ...

  8. java统计文本中单词出现的个数

    package com.java_Test; import java.io.File; import java.util.HashMap; import java.util.Iterator; imp ...

  9. Unity Shader 菲涅尔环境反射

    菲涅尔反射描述了一种光学现象,当光照到物体表面时,一部分发生反射,另一部分则进入物体内部,发生折射或散射:相比直接的反射和折射计算,菲涅尔反射更接近真实情况. 可用下面的等式近似计算这种反射效果: F ...

  10. JS工具整理

    1.获取今日日期:摘抄地址:https://www.cnblogs.com/carekee/articles/1678041.html getTodayFmt('yyyy-MM-dd') getTod ...