这题听说是道十分经典的剪枝算的题目,不要问我剪枝是什么,我也不知道,反正我只知道用到了深度搜索

我参考了好多资料才悟懂,然后我发现网上的那些大神原理讲的很明白,但代码没多少注释,看的很懵X,于是我抄起VS写了个详细注释版,真的很详细,史上最详细,全宇宙最详细,就这么自信,不信你看,看不懂你咬我。

/*--------------------------------------------
* 剪枝算法经典例题Sticks详细注释版
*---------------------------------------------*/ #include <iostream> using namespace std; //某根木棍被使用过就设置标志位为1,没使用过则设置标志位为0
#define USED 1
#define UNUSED 0
#define MAXSIZE 64 //定义木棍结构体
typedef struct Stick
{
int length;//每根木棍的长度
int mark;//标志位,记录是否被使用过
}Sticks[MAXSIZE]; //虽然程序中使用全局变量是不好的习惯,但做题who care
int n;//未拼接时的总木棒数
int g;//拼接好后的木棒数
int len;//满足要求的拼接后的木棒的长度
Sticks sticks;//定义一个木棒集合,记录未拼接时各木棒的信息 //定义一个冒泡排序算法,从大到小排序
void BubbleSort(Sticks *a,int num)
{
int i,j;
int temp;//交换时用的中间变量
for (i = 0;i<num-1;i++)
{
for (j = 0;j<num-1;j++)
{
if ((*a)[j].length<(*a)[j+1].length)
{
//交换
temp = (*a)[j].length;
(*a)[j].length = (*a)[j+1].length;
(*a)[j+1].length = temp;
}
}
}
} /*剪枝函数,深度搜索
*总共三个参数,nowLen表示现在拼接成的木棒的长度,
*nowGet表示现在拼接成的木棒的总数,此值若等于之前的g则说明找到符合要求的木棒长,
*cnt表示拼接过程中查找剩下符合要求的木棒从哪个下标开始查找,当cnt大于n时说明没有符合要求的拼接方法
*该函数返回1代表找到了,0代表没有找到符合要求的*/
int DFS(int nowLen,int nowGet,int cnt)
{
if (cnt>=n) return 0;//找的下标都超了,肯定不满足
if (nowGet == g) return 1;//如果这个长度下获取的总个数与g相等说明符合条件
int i; //开始遍历查找
for (i = cnt;i<n;i++)
{
if (sticks[i].mark==UNUSED)
{
//当找到的一组木棒恰好能拼接成需要的长度时
if (nowLen+sticks[i].length == len)
{
sticks[i].mark = USED;//设置这个木棒已使用过
//这组满足,开始下一组寻找
if (DFS(0,nowGet+1,nowGet)==1)
{
//递归,直到最后每一组都满足需要的长度时说明这个长度可行
return 1;
}
sticks[i].mark = UNUSED;//解除使用
return 0;
}
//当找到的一组木棒还小于拼接成需要的长度时
else if(nowLen+sticks[i].length < len)
{
sticks[i].mark = USED;//设置这个木棒已使用过
if(DFS(nowLen+sticks[i].length,nowGet,i+1)==1)
{
//同样递归,这里说明一下i+1,这个意思是从i+1下标开始寻找要使这组木棒满足要求的木棒
return 1;
}
sticks[i].mark = UNUSED;//解除使用
//下面这句表示如果当前搜索时,前边的长度为0,而第一根没有成功的使用,
//说明第一根始终要被废弃,所以这种组合必定不会成功
//此处的剪枝必须有,因为这里的剪枝会节省很多的无用搜索,
//缺少这一句超时
if (nowLen == 0) return 0;
//下面这句是指如果有一根木棒加上去已经知道不满足要求了,则与它相同长度的木棒都可以跳过
for ( ;sticks[i].length==sticks[i+1].length&&i+1<n;i++);
}
} } return 0;
} int main()
{
int i;
int sum;//木棒的总长 //用户输入每组的木棒数
while(cin>>n,n)
{
//木棒一开始都初始化为未使用过的
for (i = 0;i<MAXSIZE;i++)
{
sticks[i].mark = UNUSED;
}
sum = 0;
for (i = 0;i<n;i++)
{
cin>>sticks[i].length;
sum+=sticks[i].length;
} //从大到小排序
BubbleSort(&sticks,n);
for (len = sticks[0].length;len<=sum;len++)
{
if (sum%len!=0) continue;//最后选的木棒长度一定是能被总木棒长整除的 g = sum/len;//拼接的后的木棒数
//剪枝,满足要求退出循环
if(DFS(0,0,0))
{
break;
}
}
//输出满足要求的木棒长
cout << len<<endl;
} return 0;
}

经典剪枝算法的例题——Sticks详细注释版的更多相关文章

  1. 【强连通分量】 Kosaraju和Tarjan算法 (标准模板+详细注释)

    codevs 题意:求最大强连通分量的大小以及所包含的顶点有哪些 Tarjan算法 #include<iostream> #include<queue> #include< ...

  2. SSM+shiro,所有配置文件,详细注释版,自用

    spring配置文件applicationContext.xml,放在resources下 <?xml version="1.0" encoding="UTF-8& ...

  3. SSM+shiro及相关插件的整合maven所有依赖,详细注释版,自用,持续更新

    整合了SSM+shiro框架,slf4j+logback日志,及一些好用的插件PageHelper,mybatis-generator,Lombok,fastjson等等 <?xml versi ...

  4. DRF 简单使用(详细注释版)

    1.djangorestframework使用 下载安装 pip install djangorestframework ## djangorestframework pip install djan ...

  5. Light OJ - 1026 - Critical Links(图论-Tarjan算法求无向图的桥数) - 带详细注释

     原题链接   无向连通图中,如果删除某边后,图变成不连通,则称该边为桥. 也可以先用Tajan()进行dfs算出所有点 的low和dfn值,并记录dfs过程中每个 点的父节点:然后再把所有点遍历一遍 ...

  6. 【蓝桥杯/算法训练】Sticks 剪枝算法

    剪枝算法 大概理解是通过分析问题,发现一些判断条件,避免不必要的搜索.通常应用在DFS 和 BFS 搜索算法中:剪枝策略就是寻找过滤条件,提前减少不必要的搜索路径. 问题描述 George took ...

  7. 紫书 例题8-17 UVa 1609 (构造法)(详细注释)

    这道题用构造法, 就是自己依据题目想出一种可以得到解的方法, 没有什么规律可言, 只能根据题目本身来思考. 这道题的构造法比较复杂, 不知道刘汝佳是怎么想出来的, 我想的话肯定想不到. 具体思路紫书上 ...

  8. acm常见算法及例题

    转自:http://blog.csdn.net/hengjie2009/article/details/7540135 acm常见算法及例题  初期:一.基本算法:     (1)枚举. (poj17 ...

  9. 【十大经典数据挖掘算法】AdaBoost

    [十大经典数据挖掘算法]系列 C4.5 K-Means SVM Apriori EM PageRank AdaBoost kNN Naïve Bayes CART 1. 集成学习 集成学习(ensem ...

随机推荐

  1. win10家庭版启用远程桌面

    此电脑右键属性->远程设置->允许远程协助连接这台计算机 勾选 下载RDP Wrapper 地址:https://github.com/stascorp/rdpwrap/releases ...

  2. PG-跨库操作-postgres_fdw

    接上一篇<PG-跨库操作-dblink>:讲下postgres_fdw的使用:postgres_fdw工作原理详细介绍可以去看下<PostgreSQL指南>第4章: 对FDW特 ...

  3. 你没有看错,爬网页数据,C# 也可以像 Jquery 那样

    一:背景 1. 讲故事 前段时间搞了一个地方性民生资讯号,资讯嘛,都是我抄你的,你抄官媒的,小市民都喜欢奇闻异事,所以就存在一个需求,如何去定向抓取奇闻异事的地方号上的新闻,其实做起来很简单,用逻辑回 ...

  4. 容器云平台No.10~通过gogs+drone+kubernetes实现CI/CD

    什么是CI/CD 持续集成(Continous Intergration,CI)是一种软件开发实践,即团队开发成员经常集成它们的工作,通常每个成员每天至少集成一次,也就意味着每天可能会发生多次集成.每 ...

  5. Java泛型中的类型参数和通配符类型

    类型参数 泛型有三种实现方式,分别是泛型接口.泛型类.泛型方法,下面通过泛型方法来介绍什么是类型参数. 泛型方法声明方式:访问修饰符 <T,K,S...> 返回类型 方法名(方法参数){方 ...

  6. 摄像头Sensor 图像格式

    以0V7725为例: 顺便介绍一下0V7725的主要管脚,管脚定义能体现功能,体现使用方法.

  7. 《C++primerplus》第12章“队列模拟”程序

    这个程序刚开始学有很多难点,个人认为主要有以下三项: 1.链表的概念 2.如何表示顾客随机到达的过程 3.程序执行时两类之间的关系,即执行逻辑 关于第一点,书上的图解释得比较清楚了,把"空指 ...

  8. 【题解】PTA-Little Bird

    Link 单调队列板子. 题目大意:一个点可以由距离它不超过\(k\)的点跳过来,如果那个点比它高就不需要花费体力,否则花费\(1\)的体力.问走到\(n\)的最小体力,多组询问. 显然的转移方程,设 ...

  9. Solon集成(02)- 轻松吃下小馒头 Dubbo

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  10. OracleOggan安装并测试同步数据步骤!

    Oracle Golden Gate (ogg)安装使用说明 Golden Gate(简称OGG)提供异构环境下交易数据的实时捕捉.变换.投递等功能. OGG支持的异构环境有: OGG的特性: ①对生 ...