简要题意

木材店老板给出一个整数 \(m\) 和 \(m\) 个木板的长度。老板给出的木板可以随意无损耗切割。

约翰给出一个整数 \(n\) 和所需要的 \(n\) 个木板的长度。

求约翰能得到最多木板的个数。

分析

二分答案的题基本不会是单纯的二分,一般会带上一些优化、DP、搜索等……

这道题主要是运用搜索和剪枝,而且剪枝不止一个。

正常搜索肯定都会写,不过优化比较多。

优化一:贪心

既然木材店的木板都可以任意分割,那么将给出的 m 个木板排序,优先使用长木板。

而对于需要的木板来说,优先使用短木板才能使木板数量最大。

优化二:前缀和

用前缀和维护所需木板长度,记录给出木板总长度。

二分前判断所需木板总长度是否已经大于给出木板总长度,超过就向回退,直到木板足够。

优化三:剪枝一

记录浪费的木材数量。一块长木板被切割后,不使用的那段就可能被浪费。

搜索过程中,如果浪费掉的木材长度与所需木材的长度超过了给出木材总长度,则不继续向下搜索。

优化四:剪枝二

如果需要的两块相邻木板长度相等,我们就可以从上一次放的那一块枚举,可以节省大量计算。

Code

来自 Trump 的 Code

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring> using namespace std; int n,m,g_wo[1145140],n_wo[1145140],pre[1145140];
int tot,mid,waste; bool check( int num , int le ) {
//num 还需要的木材数量 le 左边界
if ( num == 0 ) return true;//全可以取到则返回true
if ( pre[mid] + waste > tot ) {
//优化3
//若需要的木板长度和浪费长度加起来超过老板提供总长度
return false;
}
for ( int i = le; i <= n; ++i) {
if ( g_wo[i] >= n_wo[num] ) {
bool k;
//g_wo[i]的长度大 可以锯下来n_wo[num]的长度
g_wo[i] -= n_wo[num];
//则先锯下来 方便向前遍历 一会儿回溯
if ( g_wo[i] < n_wo[1] ) {
//剩下的木材若连第一个(最短的)需要木板的长度都没有
waste += g_wo[i];//则是无法使用的废料
}
if ( n_wo[num] == n_wo[num - 1] ) {
//优化4
k = check( num - 1 , i );
//num - 1只用从i开始搜就行
//因为是降序排列,所以num - 1一定小于等于num表示的长度
//若num前一个与num需要木材长度一样
//则num - 1的长度一定大于i之前的提供木板的长度
//所以直接从i开始向后找
}
else k = check( num - 1 , 1 );
if ( g_wo[i] < n_wo[1] ) {
waste -= g_wo[i];
//回溯
//前面因为是先减n_wo[1]再判断
//所以这里的判断要先进行
}
g_wo[i] += n_wo[num];//回溯
if ( k == true ) return true;
//入果已经有可能则直接返回ture
}
}
return false;
} int main() {
scanf("%d",&m);
for (int i = 1; i <= m; ++i) {
scanf("%d",&g_wo[i]);//提供的木材
tot += g_wo[i];
}
scanf("%d",&n);
for (int i = 1; i <= n; ++i) {
scanf("%d",&n_wo[i]);//需要的木材
}
sort( g_wo + 1 , g_wo + 1 + m );
sort( n_wo + 1 , n_wo + 1 + n );
//优化1
for (int i = 1; i <= n; ++i) {
pre[i] = n_wo[i];
pre[i] += pre[i - 1];//前缀和
}//优化2
int l_an = 0,r_an = n;//表示最多的木板数
while ( r_an > 0 && pre[r_an] > tot ) {
r_an --;//若需要的长度大于提供长度
}
while ( l_an <= r_an ) {//正常二分
waste = 0;//记得初始化QWQ
mid = l_an + ( ( r_an - l_an ) >> 1 );
if ( check( mid , 1 ) ) l_an = mid + 1;
else r_an = mid - 1;
}
//输出r是因为在l = r的情况下,若mid可取,则l会变成mid+1但r不变
//若不可取则r会向前-1
printf("%d\n",r_an);
//二分答案一般二分的都是让输出的内容
return 0;
}

总结

总的来说,这道题是以搜索为辅的二分答案,重点是优化很多。

经过一段时间思考想出了前两个优化,优化三四没想出来。

这道题提供了一种新的类型的节省计算的方法,还是很有价值的。

[SCOI2005] 栅栏【题解】的更多相关文章

  1. bzoj 1082: [SCOI2005]栅栏 题解

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

  2. BZOJ1082: [SCOI2005]栅栏 题解

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

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

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

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

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

  5. 洛谷 P2329 [SCOI2005]栅栏 解题报告

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

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

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

  7. SCOI2005栅栏

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

  8. bzoj1082[SCOI2005]栅栏

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

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

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

  10. bzoj 1082: [SCOI2005]栅栏

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

随机推荐

  1. ChatCLM部署随笔

    ChatCLM 博客 ChatGLM Github ChatGLM-webui 介绍 ChatGLM-6B 是一个开源的.支持中英双语的对话语言模型,基于 General Language Model ...

  2. 群论中的 Lagrange 定理

    今天跟 hym 打球时讲到了这个东西,突然发现证明拉格朗日定理的思想有许多跟轨道-稳定集定理很像,所以这里又记录一下. 为了证明 Lagrange 定理,我们需要了解一些关于子群和陪集的性质. 首先给 ...

  3. 通过Handsontable实现像Excel一样编辑数据

    ​一.Handsontable是指什么? 官网: http://handsontable.com Handsontable是一个JavaScript库,可以帮助您轻松实现类似Excel电子表格一样的编 ...

  4. "树形List"与"扁平List"互转(Java实现)

    背景:在平时的开发中,我们时常会遇到下列场景 公司的组织架构的数据存储与展示 文件夹层级的数据存储与展示 评论系统中,父评论与诸多子评论的数据存储与展示 ...... 对于这种有层级的结构化数据,就像 ...

  5. 如何理解 Spring Boot 中的 Starter ?

    假如 没有 Spring Boot Starter,我们有两种方式来创建 Spring Bean. spring xml 模式 (远古模式,并不推荐) spring API 来创建 Spring Be ...

  6. vue下载附件按钮功能

    一.tools文件夹下tools文件中封装下载方法: const iframeId = 'download_file_iframe' function iframeEle (src) { let el ...

  7. 2020-12-06:mysql中,多个索引会有多份数据吗?

    福哥答案2020-12-06: 数据不会有多份,索引有几个就有几份.聚簇索引存数据和索引,非聚簇索引存索引,聚簇索引只有一个,非聚簇索引可以有多个.

  8. 2021-02-26:一个数组arr是二叉树的中序遍历结果,每条边的开销是父节点和子节点的乘积,总开销是所有边的开销之和。请问最小总开销是多少?

    2021-02-26:一个数组arr是二叉树的中序遍历结果,每条边的开销是父节点和子节点的乘积,总开销是所有边的开销之和.请问最小总开销是多少? 链接:https://www.nowcoder.com ...

  9. 2022-04-12:给定一个字符串形式的数,比如“3421“或者“-8731“, 如果这个数不在-32768~32767范围上,那么返回“NODATA“, 如果这个数在-32768~32767范围上

    2022-04-12:给定一个字符串形式的数,比如"3421"或者"-8731", 如果这个数不在-32768~32767范围上,那么返回"NODAT ...

  10. GPT大语言模型Alpaca-lora本地化部署实践【大语言模型实践一】

    模型介绍 Alpaca模型是斯坦福大学研发的LLM(Large Language Model,大语言)开源模型,是一个在52K指令上从LLaMA 7B(Meta公司开源的7B)模型微调而来,具有70亿 ...