【洛谷UVA307】小木棍Sticks
算法的话:dfs+超强剪枝;
(另外注意UVA上好像不接受万能头[因为万能头WA了两次,瑟瑟发抖])
思路:
最直接的思路,枚举木棍长度来dfs,但这样很容易就TLE了。
dfs的四项关键字:
1.num:剩多少切割后的木棍没有使用;
2.rest:还需要rest的长度可以凑成一个切割前的木棍;
3.len:切割前的木棍长度;
4.last:当前切割后的木棍起始的下标;(对于初始状态dfs last是1还是0好迷啊,好像初始状态是1或者0都可以ac???) 行吧就当我没说亲测不行qwq目标条件:num=0且rest=0;
剪枝1:对于这些木棍来说,无论怎样分,若满足条件,那么木棍长度一定是总和的倍数;
剪枝2:搜索上下界:上界:所有木棍的长度和,下界:最长一根木棍的长度;
剪枝3:将切割后的木棍按从大到小排列,对于一根切割后木棍,如果它不能满足拼成一根长度为len的木棍,那么它之前的比它长的也显然不行,因此last的作用就是减少枚举次数,每次从上一次的last+1枚举就可以了;
剪枝4:如果一根木棍的长度>rest,那么对于每一根长度与之相同的木棍都不可能组合,直接跳过减少枚举次数;
剪枝5:如果当前切割后的木棍长度=len,但组装失败(如果成功就已经return true了)即返回false,因为从大到小排序,当前的切割后木棍明明可以组成成功,但后面的切割后的木棍却组成失败,说明后面的切割后的木棍长度不够或者直接构不成(剩下的切割后的任意长度和!=切割前的木棍长度),那么继续搜肯定不行;????
#include <iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int n, mg[], tmp, top = , tt = , vis[],T; int cmp(int a, int b) {
return a > b;
}
bool dfs(int num , int len , int rest , int last ) {
if (rest == && num == )
return ;
if (num == )//没有木棍,不能拼成的情况
return ;
if (rest == ) {//如果现在刚好组成一根木棍,更新rest变为枚举的切割前木棍的长度
rest = len;
last = ;//last再次更新为0
}
for (int i = last + ; i < top; i++) { //剪枝3
if (vis[i])
continue;
if (mg[i] > rest) {
while (mg[i] == mg[i + ] && i < top) i++; //剪枝4
continue;
}
vis[i] = ;
if (dfs(num - , len, rest - mg[i], i))//选择这根木棍后继续递归,判断是否可以构成一根木棍
return ;
vis[i] = ;
if ((mg[i] == rest) || (len == rest))//剪枝5:这句话亲测对于提速还是很有效的
break;
while (mg[i] == mg[i + ] && i < top) i++;
}
return ;
}
int main() {
while(){//while循环来输入多组数据
memset(mg,,sizeof(mg));//因为处理多组数据,所以记得清空数组
memset(vis,,sizeof(vis));
tt = ;top = /*count number*/;
cin>>n;
if(n == ) return ;
for (int i = ; i <= n; i++) {
cin >> tmp;
mg[top++] = tmp,/*存到数组里*/ tt += tmp;//求和
}
sort(mg + , mg + top, cmp);//从大到小排序
for (int i = mg[]; i <= tt; i++) {//剪枝2
if (tt % i != )//剪枝1
continue;
if (dfs(top - /*因为最后top会在这n个数的基础上+1,故dfs起点为top-1*/, i, i, )) {//if this i is ok to make sticks,just choose it and break;
cout << i <<endl;
break;
}
}
} return ;
}
立下凌云壮flag,咱今天绝对不理yy,咱不烦yy,咱不让yy烦
end-
【洛谷UVA307】小木棍Sticks的更多相关文章
- 洛谷P1120 小木棍
洛谷1120 小木棍 题目描述 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过50. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它们的长 ...
- 洛谷P1120 小木棍 [数据加强版](搜索)
洛谷P1120 小木棍 [数据加强版] 搜索+剪枝 [剪枝操作]:若某组拼接不成立,且此时 已拼接的长度为0 或 当前已拼接的长度与刚才枚举的长度之和为最终枚举的答案时,则可直接跳出循环.因为此时继续 ...
- 洛谷P1120 小木棍(sticks数据加强版)
题目描述 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过50. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它们的长度. 给出每段小木棍的长度,编 ...
- 【题解】洛谷P1120 小木棍(搜索+剪枝+卡常)
洛谷P1120:https://www.luogu.org/problemnew/show/P1120 思路 明显是搜索题嘛 但是这数据增强不是一星半点呐 我们需要N多的剪枝 PS:需要先删去超出50 ...
- 洛谷P1120小木棍[DFS]
题目描述 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过50. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它们的长度. 给出每段小木棍的长度,编 ...
- 洛谷 P1120 小木棍 [数据加强版]解题报告
P1120 小木棍 [数据加强版] 题目描述 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过50. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它 ...
- [UVA307]小木棍 Sticks
题目大意:有一堆小木棍,把它们接成相同长度的小木棍,问结果的小木棍的最小长度是多少,多组数据 题解:$dfs$,各种剪枝. 卡点:无 C++ Code: #include <cstdio> ...
- 洛谷P1120 小木棍 [搜索]
题目传送门 题目描述乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过50. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍和它们的长度. 给出每段小木棍 ...
- [洛谷P1120]小木棍 [数据加强版]
题目大意:有一些同样长的木棍,被切割成几段(长$\leqslant$50).给出每段小木棍的长度,找出原始木棍的最小可能长度. 题解:dfs C++ Code: #include<cstdio& ...
- 洛谷——P1120 小木棍 [数据加强版]
P1120 小木棍 [数据加强版] 题目描述 乔治有一些同样长的小木棍,他把这些木棍随意砍成几段,直到每段的长都不超过5050. 现在,他想把小木棍拼接成原来的样子,但是却忘记了自己开始时有多少根木棍 ...
随机推荐
- ReactiveCocoa 之 优雅的 RACCommand
RACCommand 是一个在 ReactiveCocoa 中比较复杂的类,大多数使用 ReactiveCocoa 的人,尤其是初学者并不会经常使用它. 在很多情况下,虽然使用 RACSignal 和 ...
- TensorFlow——MNIST手写数字识别
MNIST手写数字识别 MNIST数据集介绍和下载:http://yann.lecun.com/exdb/mnist/ 一.数据集介绍: MNIST是一个入门级的计算机视觉数据集 下载下来的数据集 ...
- 搭建Keepalived+LNMP架构web动态博客 实现高可用与负载均衡
环境准备: 192.168.193.80 node1 192.168.193.81 node2 关闭防火墙 [root@node1 ~]# systemctl stop firewalld #两台都 ...
- 02tensorflow非线性回归以及分类的简单实用,softmax介绍
import tensorflow as tf import numpy as np import matplotlib.pyplot as plt # 使用numpy生成200个随机点 x_data ...
- sqlmap 基本使用步骤(一)
列出数据据信息:python sqlmap.py -u "http://ctf5.shiyanbar.com/web/index_3.php?id=1" --dbs 列出当前数据库 ...
- web前后端分离漏洞分析防御
web前后端分离漏洞分析防御 漏洞分析,主要漏洞有 一.跨站脚本攻击XSS 程序 + 数据 = 结果:攻击后,数据夹杂一部分程序(执行代码),导致结果改变: 1.XSS攻击注入点 (a):HTML节点 ...
- CodeChef Max-digit Tree(动态规划)
传送门. 题解: 最主要的问题是如何判断一个数是否合法,这就需要发现性质了. 这个状态划分还是不太容易想到, 每次加的数\(∈[0,k)\),也就是个位一直在变变变,更高的位每次都是加一,这启发我们状 ...
- 小米笔记本pro版bios经常找不到硬盘
自从买了小米笔记本,对小米的印象大大折扣,bios经常找不到硬盘,关机,重启,就好了. 到小米售后,售后说是系统坏了,我说bios里都找不到.他说,系统坏了也会出现这个情况.我说好吧.重做后,没用几天 ...
- PL/SQL 循环
----PL/SQL基本循环语句 LOOP DECLARE x ; BEGIN LOOP dbms_output.put_line(x); x :; THEN exit; END IF; END LO ...
- vim 批量添加删除注释
vim中单行注释只是多行注释的一个特例,这里统一进行多行注释的讲解 (1)添加批量注释 ctrl+v 进入列编辑模式,向下或向上移动光标,把需要注释的行的开头标记起来,然后按大写的I(shift+i) ...