cn域名做外贸网站,共和网站建设公司,移动网站建设制作,注册网络科技公司需要什么条件1、回溯法 (1)描述:回溯法是一种选优搜索法#xff0c;按选优条件向前搜索#xff0c;以达到目标。但当探索到某一步时#xff0c;发现原先选择并不优或达不到目标#xff0c;就退回一步重新选择#xff0c;这种走不通就退回再走的技术为回溯法。 (2)原理: 回溯法在问题的… 1、回溯法 (1)描述:回溯法是一种选优搜索法按选优条件向前搜索以达到目标。但当探索到某一步时发现原先选择并不优或达不到目标就退回一步重新选择这种走不通就退回再走的技术为回溯法。 (2)原理: 回溯法在问题的解空间树中按深度优先策略从根结点出发搜索解空间树。算法搜索至解空间树的任意一点时先判断该结点是否包含问题的解。如果肯定不包含则跳过对该结点为根的子树的搜索逐层向其祖先结点回溯否则进入该子树继续按深度优先策略搜索。 回溯法的基本做法是搜索或是一种组织得井井有条的能避免不必要搜索的穷举式搜索法。这种方法适用于解一些组合数相当大的问题。有许多问题当需要找出它的解集或者要求回答什么解是满足某些约束条件的最佳解时往往要使用回溯法。 (3)问题的解空间 问题的解向量回溯法希望一个问题的解能够表示成一个n元式(x1,x2,…,xn)的形式。 显约束对分量xi的取值限定。 隐约束为满足问题的解而对不同分量之间施加的约束。 解空间对于问题的一个实例解向量满足显式约束条件的所有多元组构成了该实例的一个解空间。 注意同一个问题可以有多种表示有些表示方法更简单所需表示的状态空间更小存储量少搜索方法简单。 (4)生成问题状态的基本方法 扩展结点:一个正在产生儿子的结点称为扩展结点。 活结点:一个自身已生成但其儿子还没有全部生成的节点称做活结点。 死结点:一个所有儿子已经产生的结点称做死结点。 深度优先的问题状态生成法如果对一个扩展结点R一旦产生了它的一个儿子C就把C当做新的扩展结点。在完成对子树C以C为根的子树的穷尽搜索之后将R重新变成扩展结点继续生成R的下一个儿子如果存在。 宽度优先的问题状态生成法在一个扩展结点变成死结点之前它一直是扩展结点。 回溯法为了避免生成那些不可能产生最佳解的问题状态要不断地利用限界函数(bounding function)来处死那些实际上不可能产生所需解的活结点以减少问题的计算量。具有限界函数的深度优先生成法称为回溯法。 (5)回溯法的基本思想 基本思想 用回溯法解题的一个显著特征是在搜索过程中动态产生问题的解空间。在任何时刻算法只保存从根结点到当前扩展结点的路径。如果解空间树中从根结点到叶结点的最长路径的长度为h(n)则回溯法所需的计算空间通常为O(h(n))。而显式地存储整个解空间则需要O(2h(n))或O(h(n)!)内存空间。 解题步骤 1)针对所给问题定义问题的解空间 2)确定易于搜索的解空间结构 3)以深度优先方式搜索解空间并在搜索过程中用剪枝函数避免无效搜索。 常用剪枝函数用约束函数在扩展结点处剪去不满足约束的子树用限界函数剪去得不到最优解的子树。 递归回溯 回溯法对解空间作深度优先搜索因此在一般情况下用递归方法实现回溯法。 迭代回溯 采用树的非递归深度优先遍历算法可将回溯法表示为一个非递归迭代过程。 子集树当所给的问题是从n个元素的集合S中找出满足某种性质的子集时相应的解空间称为子集树。例如那个物品的0-1背包问题所相应的解空间树就是一颗子集树。这类子集问题通常有2^n个叶节点其节点总个数为2^(n1)-1。遍历子集树的任何算法均需要O(2^n)的计算时间。 用回溯法遍历子集树的一般算法可描述如下 排列树当所给问题是确定n个元素满足某种性质的排列时相应的解空间树称为排列树。排列树通常有n!个叶子节点。因此遍历排列树需要O(n!)的计算时间。 用回溯法遍历排列树的一般算法可描述如下 问题描述有一批共n个集装箱要装上2艘载重量分别为c1和c2的轮船其中集装箱i的重量为wi且装载问题要求确定是否有一个合理的装载方案可将这些集装箱装上这2艘轮船。如果有找出一种装载方案。 例如当n3,c1c250,且w[10,40,40]时则可以将集装箱1和2装到第一艘轮船上而将集装箱3装到第二艘轮船上如果w[20,40,40]则无法将这3个集装箱都装上轮船。 基本思路 容易证明如果一个给定装载问题有解则采用下面的策略可得到最优装载方案。 (1)首先将第一艘轮船尽可能装满 (2)将剩余的集装箱装上第二艘轮船。 将第一艘轮船尽可能装满等价于选取全体集装箱的一个子集使该子集中集装箱重量之和最接近C1。由此可知装载问题等价于以下特殊的0-1背包问题。 用回溯法设计解装载问题的O(2^n)计算时间算法。在某些情况下该算法优于动态规划算法。 算法设计 用回溯法解装载问题时用子集树表示其解空间显然是最合适的。用可行性约束函数可剪去不满足约束条件的子树。在子集树的第j1层的结点z处用cw记当前的装载重量即cw则当cwc1时以结点z为根的子树中所有结点都不满足约束条件因而该子树中的解均为不可行解故可将该子树剪去。该约束函数去除不可行解得到所有可行解。 可以引入一个上界函数用于剪去不含最优解的子树从而改进算法在平均情况下的运行效率。设z是解空间树第i层上的当前扩展结点。cw是当前载重量bestw是当前最优载重量r是剩余集装箱的重量即r。定义上界函数为cwr。在以z为根的子树中任一叶结点所相应的载重量均不超过cwr。因此当cwrbestw时可将z的右子树剪去。 #include stdafx.h #include iostream using namespace std; templateclass Type Type MaxLoading(Type w[ ], Type c, int n, int bestx[ ]); int main() { int n3,m; int c50,c250; int w[4]{0,10,40,40}; int bestx[4]; mMaxLoading(w, c, n, bestx); cout轮船的载重量分别为endl; coutc(1)c,c(2)c2endl; cout待装集装箱重量分别为endl; coutw(i); for (int i1;in;i) { coutw[i] ; } coutendl; cout回溯选择结果为endl; coutm(1)mendl; coutx(i); for (int i1;in;i) { coutbestx[i] ; } coutendl; int m20; for (int j1;jn;j) { m2m2w[j]*(1-bestx[j]); } coutm(2)m2endl; if(m2c2) { cout因为m(2)大于c(2),所以原问题无解endl; } return 0; } template class Type Type MaxLoading(Type w[],Type c,int n,int bestx[])//迭代回溯法返回最优载重量及其相应解初始化根结点 { int i1;//当前层x[1:i-1]为当前路径 int *xnew int[n1]; Type bestw0, //当前最优载重量 cw0, //当前载重量 r0; //剩余集装箱重量 for (int j1;jn;j) { rw[j]; } while(true)//搜索子树 { while(in cww[i]c)//进入左子树 { r-w[i]; cww[i]; x[i]1; i; } if (in)//到达叶结点 { for (int j1;jn;j) { bestx[j]x[j]; } bestwcw; } else//进入右子树 { r-w[i]; x[i]0; i; } while (cwrbestw) { //剪枝回溯 i--; while (i0 !x[i]) { rw[i]; i--; } //从右子树返回 if (i0) { delete []x; return bestw; } x[i]0; cw-w[i]; i; } } } 转载于:https://www.cnblogs.com/yuxingxing/p/5500168.html