[bzoj1082][SCOI2005]栅栏(二分+dfs剪枝)

该博客详细介绍了如何解决SCOI2005年的栅栏问题,利用二分查找策略配合深度优先搜索(DFS)进行剪枝。在题解中,博主提出了几点优化措施,包括对两个数组排序,从大到小枚举所需的木板,从小到大枚举提供的木板,并在搜索过程中考虑浪费的木板部分。此外,还讨论了等长木板的情况以及通过判断a和len的关系来进一步优化搜索范围。最后,提供了实现这些策略的代码。
摘要由CSDN通过智能技术生成

题目:

我是超链接

题解:

二分,我们可以发现砍出来的都是前mid小的。
那么剩下的就是判断这mid个能否被截出来?
判断用dfs,虽然范围不大你也不能这么暴力求解啊
考虑(奇奇怪怪的)优化!
1、将两个数组分别排序,那么所需的木板一定是从大到小枚举,同时枚举每一个需要的木板时,按照提供的木板的大小从小到大枚举能不能切去(贪心
2、对于一块提供的木板,如果其(切后||没切)的长度小于需要的最小的木板那么剩下的部分一定浪费。记录一个waste表示浪费的部分,如果waste+所需木板总长度>提供木板总长度,直接回溯
3、还有一些优化,比如说如果两块木板等长的话,你可以在下次搜索的时候从这次的i木板开始
4、其实还有一些“强行”优化,比如说你可以通过判断a和len的关系给l,r缩进(博主没有写╮(╯▽╰)╭

代码:

#include <cstdio>
#include <algorithm>
#include <iostream>
#include <cstring>
#define N 1005
using namespace std;
int have[N],want[N],all,sum[N],waste,m;
bool fff;
int cmp(int a,int b){return a>b;}
void dfs(int t,int w,int now)
{
    if (fff) return;
    if (t==0){fff=true; return;}
    for (int i=now;i<=m;i++)
      if (want[t]<=have[i] && !fff)
      {
        have[i]-=want[t];
        int lj=w;
        if (have[i]<want[1]) lj+=have[i];
        if (lj>waste) {have[i]+=want[t];continue;}
        if (want[t]==want[t-1]) now=i;else now=1;
        dfs(t-1,lj,now);
        have[i]+=want[t];
      }
}
bool check(int mid)
{
    fff=0;dfs(mid,0,1);
    return fff;
}
int main()
{
    int n,i;
    scanf("%d",&m);
    for (i=1;i<=m;i++) scanf("%d",&have[i]),all+=have[i];
    sort(have+1,have+m+1,cmp);
    scanf("%d",&n);
    for (i=1;i<=n;i++) scanf("%d",&want[i]);
    sort(want+1,want+n+1);
    for (i=1;i<=n;i++) sum[i]=sum[i-1]+want[i];
    int l=1,r=n,ans=0;
    while (l<=r)
    {
        int mid=(l+r)>>1;
        waste=all-sum[mid];
        if (check(mid)) l=mid+1,ans=mid;
        else r=mid-1;
    }
    printf("%d",ans);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值