搜索+剪枝——POJ 1011 Sticks
搜索+剪枝——POJ 1011 Sticks
博客分类:
- 算法
非常经典的搜索题目,第一次做还是暑假集训的时候,前天又把它翻了出来
本来是想找点手感的,不想在原先思路的基础上,竟把它做出来了而且还是0ms过得。仔细想想,对搜索又有了一点点认识。
题目要求将一系列的sticks重新组合,形成若干相等相等长度的木棒,且尽量使木棒长度最小,如果数据量比较小的话,就纯粹是搜索了,但题目要求的 sticks可能达到64根,如果纯粹的搜索则显然是会远远超过1000ms的,因而也就把剪枝放在了很重要的位置。从第一根stick开始,寻找下一根 stick使两者的长度小于等于木棒的长度,然后再寻找下一根stick,直到和为一根木棒的长度;然后又从一根没有被使用的stick开始进行下一根木棒的组合。概括一点说,就是一个深度优先搜索。
做完这个题目之后,仔细想了想,觉得之前没有做出来最主要的原因就出在回溯上,因为对于dfs最初使用的的返回类型是void,当某种情况不行,需要回溯到上一层时,我就会很急地使用return语句,这个时候并没有回溯到上一层,而是退出了函数;另外,也要告诫自己要控制好变量,就这个题目而言,开始也有地方的那个used值并没有做好处理。致使最后有时候会出现一些莫名其妙的结果,与产生错误的提示信息。也走了那条从TLE到WA,再到AC的路,但还是学到了很多东西。
- #include <iostream>
- #include <algorithm>
- using namespace std;
-
- int sticks[64], n, len, num;
- bool used[64];
-
- bool compare(int a, int b)
- {
- return a > b;
- }
-
- bool dfs(int cur, int left, int level)
- { //cur: 当前已经计算的木棒编号,left:该段还剩的长度,level:已经成功的木棒数
- if(left == 0) {//匹配一根木棒成功
- if(level == num-2)
- return true;
- for(cur = 0; used[cur]; cur++)
- ;
- used[cur] = true;
- if(dfs(cur+1, len-sticks[cur], level+1))
- return true;
- used[cur] = false;
- return false;
- } else {
- if(cur >= n-1)
- return false;
- for(int i = cur; i < n; i++) {
- if(used[i])
- continue;
- if((sticks[i] == sticks[i-1]) && !used[i-1])
- continue;
- if(sticks[i] > left)
- continue;
- used[i] = true;
- if(dfs(i, left-sticks[i], level))
- return true;
- used[i] = false;
- }
- return false;
- }
- }
-
- int main()
- {
- while(cin>>n) {
- if(n == 0)
- break;
- int sum = 0;
- for(int i = 0; i < n; i++) {
- scanf("%d", &sticks[i]);
- sum += sticks[i];
- }
- sort(sticks, sticks+n, compare); //由大到小排序
- bool end = false;
- for(len = sticks[0]; len <= sum/2; len++) {
- if(sum%len == 0) {
- used[0] = true;
- num = sum/len;
- if(dfs(0, len-sticks[0], 0)) {
- end = true;
- printf("%d\n", len);
- break;
- }
- used[0] = false;
- }
- }
- if(!end)
- printf("%d\n", sum);
- memset(used, 0, sizeof(used));
- }
- //system("pause");
- return 0;
- }
搜索+剪枝——POJ 1011 Sticks
- 博客分类:
- 算法
非常经典的搜索题目,第一次做还是暑假集训的时候,前天又把它翻了出来
- #include <iostream>
- #include <algorithm>
- using namespace std;
- int sticks[64], n, len, num;
- bool used[64];
- bool compare(int a, int b)
- {
- return a > b;
- }
- bool dfs(int cur, int left, int level)
- { //cur: 当前已经计算的木棒编号,left:该段还剩的长度,level:已经成功的木棒数
- if(left == 0) {//匹配一根木棒成功
- if(level == num-2)
- return true;
- for(cur = 0; used[cur]; cur++)
- ;
- used[cur] = true;
- if(dfs(cur+1, len-sticks[cur], level+1))
- return true;
- used[cur] = false;
- return false;
- } else {
- if(cur >= n-1)
- return false;
- for(int i = cur; i < n; i++) {
- if(used[i])
- continue;
- if((sticks[i] == sticks[i-1]) && !used[i-1])
- continue;
- if(sticks[i] > left)
- continue;
- used[i] = true;
- if(dfs(i, left-sticks[i], level))
- return true;
- used[i] = false;
- }
- return false;
- }
- }
- int main()
- {
- while(cin>>n) {
- if(n == 0)
- break;
- int sum = 0;
- for(int i = 0; i < n; i++) {
- scanf("%d", &sticks[i]);
- sum += sticks[i];
- }
- sort(sticks, sticks+n, compare); //由大到小排序
- bool end = false;
- for(len = sticks[0]; len <= sum/2; len++) {
- if(sum%len == 0) {
- used[0] = true;
- num = sum/len;
- if(dfs(0, len-sticks[0], 0)) {
- end = true;
- printf("%d\n", len);
- break;
- }
- used[0] = false;
- }
- }
- if(!end)
- printf("%d\n", sum);
- memset(used, 0, sizeof(used));
- }
- //system("pause");
- return 0;
- }
搜索+剪枝——POJ 1011 Sticks的更多相关文章
- DFS(剪枝) POJ 1011 Sticks
题目传送门 /* 题意:若干小木棍,是由多条相同长度的长木棍分割而成,问最小的原来长木棍的长度: DFS剪枝:剪枝搜索的好题!TLE好几次,终于剪枝完全! 剪枝主要在4和5:4 相同长度的木棍不再搜索 ...
- 搜索 + 剪枝 --- POJ 1101 : Sticks
Sticks Problem's Link: http://poj.org/problem?id=1011 Mean: http://poj.org/problem?id=1011&lan ...
- POJ 1011 - Sticks DFS+剪枝
POJ 1011 - Sticks 题意: 一把等长的木段被随机砍成 n 条小木条 已知他们各自的长度,问原来这些木段可能的最小长度是多少 分析: 1. 该长度必能被总长整除 ...
- POJ 1011 Sticks 【DFS 剪枝】
题目链接:http://poj.org/problem?id=1011 Sticks Time Limit: 1000MS Memory Limit: 10000K Total Submissio ...
- OpenJudge 2817:木棒 / Poj 1011 Sticks
1.链接地址: http://bailian.openjudge.cn/practice/2817/ http://poj.org/problem?id=1011 2.题目: 总时间限制: 1000m ...
- POJ 1011 Sticks(搜索 && 剪枝 && 经典)
题意 : 有n根木棍(n<=64),它们由一些相同长度的木棍切割而来,给定这n根木棍的长度,求使得原来长度可能的最小值. 分析 : 很经典的深搜题目,我们发现答案只可能是所有木棍长度总和的因数, ...
- poj 1011 Sticks (DFS+剪枝)
Sticks Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 127771 Accepted: 29926 Descrip ...
- POJ 1011 Sticks(dfs+剪枝)
http://poj.org/problem?id=1011 题意:若干个相同长度的棍子被剪成若干长度的小棍,求每根棍子原来的可能最小长度. 思路:很经典的搜索题. 我一开始各种超时,这题需要很多剪枝 ...
- POJ 1011 Sticks dfs,剪枝 难度:2
http://poj.org/problem?id=1011 要把所给的集合分成几个集合,每个集合相加之和ans相等,且ans最小,因为这个和ans只在[1,64*50]内,所以可以用dfs一试 首先 ...
随机推荐
- Kernel 3.0.8 内存管理函数【转】
转自:http://blog.csdn.net/myarrow/article/details/7208777 1. 内存分配函数 相关代码如下: #define alloc_pages(gfp_ma ...
- 命令查看WebSphere MQ运行状态
参考:https://wenku.baidu.com/view/34e40e2ffd0a79563c1e72b9.html 一.查看队列管理器运行状态 # dspmq 显示结果中QMNAME表示MQ队 ...
- CSS高度塌陷问题解决方案
高度塌陷的存在:原因分析 1 <!DOCTYPE html> <html> <head> <meta charset="utf-8" /& ...
- elasticsearch5.0.1集群一次误删除kibana索引引发的血案
elasticsearch集群中一次删除kibana索引引发的血案 1.问题发生的过程: 早上的时候有某个索引无法看到报表数据,于是就点该报表多次,估计集群被点挂了,报错:Elasticsearch ...
- vue系列之flex经典案例
案例分析: 1.中间文字居中 2.文字俩边有横线 横线无法固定宽度,因为在大屏手机上,容易出现Bug,宽度不够,俩边会出现大量空隙 解决办法,使用flex布局(网站链接) 代码: <div cl ...
- ipython+notebook使用教程(转载)
ipython是python交互环境的增强版 IPython notebook目前已经成为用Python做教学.计算.科研的一个重要工具.IPython Notebook使用浏览器作为界面,向后台的I ...
- 使用@font-family时各浏览器对字体格式(format)的支持情况
说到浏览器对@font-face的兼容问题,这里涉及到一个字体format的问题,因为不同的浏览器对字体格式支持是不一致的,这样大家有必要了解一下,各种版本的浏览器支持什么样的字体,前面也简单带到了有 ...
- 【splunk】一些查询例子
最重要资料: 入门基础:http://docs.splunk.com/Documentation/Splunk/6.5.2/SearchTutorial/WelcometotheSearchTutor ...
- system
system("cls"); //清屏 system("color f2") //改变控制台颜色 f2为颜色样式,可以是e2.f3等等 Original:htt ...
- poj3728 倍增法lca 好题!
lca的好题!网上用st表和离线解的比较多,用树上倍增也是可以做的 不知道错在哪里,等刷完了这个专题再回来看 题解链接https://blog.csdn.net/Sd_Invol/article/de ...