第5章回溯法(使用)
- 格式:pptx
- 大小:290.97 KB
- 文档页数:39
第5章回溯法回溯法也称为试探法,该方法首先暂时放弃关于问题规模大小的限制,并将问题的候选解按某种顺序逐一枚举和检验。
当发现当前候选解不可能是解时,就选择下一个候选解;倘若当前候选解除了还不满足问题规模要求外,满足所有其他要求时,继续扩大当前候选解的规模,并继续试探。
如果当前候选解满足包括问题规模在内的所有要求时,该候选解就是问题的一个解。
在回溯法中,放弃当前候选解,寻找下一个候选解的过程称为回溯。
扩大当前候选解的规模,以继续试探的过程称为向前试探。
回溯法的基本思想:确定了解空间的组织结构后,回溯法就从开始结点(根结点)出发,以深度优先的方式搜索整个解空间。
这个开始结点就成为一个活结点,同时也成为当前的扩展结点。
在当前的扩展结点处,搜索向纵深方向移至一个新结点。
这个新结点就成为一个新的活结点,并成为当前扩展结点。
如果在当前的扩展结点处不能再向纵深方向移动,则当前扩展结点就成为死结点。
换句话说,这个结点不再是一个活结点。
此时,应往回移动(回溯)至最近的一个活结点处,并使这个活结点成为当前的扩展结点。
回溯法即·96· ACM 程序设计培训教程以这种工作方式递归地在解空间中搜索,直至找到所要求的解或解空间中已没有活结点时为止。
运用回溯法解题通常包含以下三个步骤: (1)针对所给问题,定义问题的解空间; (2)确定易于搜索的解空间结构;(3)以深度优先的方式搜索解空间,并且在搜索过程中用剪枝函数避免无效搜索;5.1〖案例1〗组合问题找出从自然数1、2、……、n 中任取r 个数的所有组合。
例如n=5,r=3的所有组合为:(1)1、2、3 (2)1、2、4 (3)1、2、5 (4)1、3、4 (5)1、3、5 (6)1、4、5 (7)2、3、4 (8)2、3、5 (9)2、4、5 (10)3、4、5则该问题的状态空间为:E={(x1,x2,x3)∣xi ∈S ,i=1,2,3},其中:S={1,2,3,4,5} 约束集为:x1<x2<x3显然该约束集具有完备性。
5 回溯法回朔法即是:若在当前位置探测到一条通路则继续向前,若在当前位置探测不到一条通路则回朔侄前一位置继续探测尚未探测的反向,直到找到一条通路或探测出无通路存在为止。
定义:也叫试探法,它是一种系统地搜索问题的解的方法。
回溯算法的基本思想是:从一条路往前走,能进则进,不能进则退回来,换一条路再试。
用回溯算法解决问题的一般步骤为:一、定义一个解空间,它包含问题的解。
二、利用适于搜索的方法组织解空间。
三、利用深度优先法搜索解空间。
四、利用限界函数避免移动到不可能产生解的子空间。
问题的解空间通常是在搜索问题的解的过程中动态产生的,这是回溯算法的一个重要特性。
回溯法是一个既带有系统性又带有跳跃性的的搜索算法。
它在包含问题的所有解的解空间树中,按照深度优先的策略,从根结点出发搜索解空间树。
算法搜索至解空间树的任一结点时,总是先判断该结点是否肯定不包含问题的解。
如果肯定不包含,则跳过对以该结点为根的子树的系统搜索,逐层向其祖先结点回溯。
否则,进入该子树,继续按深度优先的策略进行搜索。
回溯法在用来求问题的所有解时,要回溯到根,且根结点的所有子树都已被搜索遍才结束。
而回溯法在用来求问题的任一解时,只要搜索到问题的一个解就可以结束。
这种以深度优先的方式系统地搜索问题的解的算法称为回溯法,它适用于解一些组合数较大的问题。
例题:1、排列的实现,对于n个不同数的排列并输出。
定义一个数组a[n],开始从第一个元素填入1开始,每进一步都是从1开始,并检查是否把n个元素填完和n个元素都没有重复的,如果是则输出一个解。
如果欲填入的数与之前已经填入的数重复,则增加1再比较,直到没有重复(除非已到了上限),如果已经到达了上限n,否回朔回上一个元素,值到其值不为上限n,然后上一个元素,下一个元素从1重新开始。
C++程序1#include<stdio.h>void main(){int n,m,i=0,j=0,k=0,flag=0,f=0;static int a[40];printf("输入N:");scanf("%d",&n);a[1]=1;k=1;while(1){flag=1;for(i=1;i<k;i++)if(a[k]==a[i]){flag=0;break;}if(flag&&k==n){for(j=1;j<=n;j++)printf("%d",a[j]);printf("\n");}if(k<n&&flag) {k++;a[k]=1;continue;}while(a[k]==n&&k>1){k--;}if(a[k]==n&&k==1)break;else{a[k]++;}}}C++程序2、排列的算法差不多,为了不重复,加了一个条件,也就是递增数列#include<stdio.h>void main(){int n,m,i=0,j=0,k=0,flag=0,f=0;static int a[40];printf("输入N,M:");scanf("%d%d",&n,&m);a[1]=1;k=1;while(1){flag=1;for(i=1;i<k;i++)if(a[k]==a[i]||a[k]<a[i])//a[k]<a[i]条件为了限制递增{flag=0;break;}if(flag&&k==m){for(j=1;j<=m;j++)printf("%d",a[j]);printf("\n");}if(k<n&&flag){k++;a[k]=1;continue;}while(a[k]==n&&k>1){k--;}if(a[k]==n&&k==1)break;else{a[k]++;}}}C++程序3#include <stdio.h>#include <string.h>void main() {int n,i=1;printf("请输入N值=?");scanf("%d",&n);int a[30]={0}; //C语言中完成数组定义并初始化不能用变量N,只能用具体数字,这儿采用比较大的30a[1]=1;while(i<=n){if (i==n){for (int j=1;j<n;j++) printf("%d ",a[j]);printf("%d\n",a[n]); //输出一组解}if (i<n) {i++;a[i]=1;continue;}while (a[i]==n && i>1) i--; //向前回溯if (a[i]==n && i==1) break; //结束else a[i]=a[i]+1;}}JA V A程序用回溯法求任意1到N个数字的全排列。