回溯法之一---算法框架及基础

系统 2021 0

回溯法其实也是一种搜索算法,它可以方便的搜索解空间。
回溯法解题通常可以从以下三步入手:
1、针对问题,定义解空间
2、确定易于搜索的解空间结构
3、以深度优先的方式搜索解空间,并在搜索的过程中进行剪枝
回溯法通常在解空间树上进行搜索,而解空间树通常有子集树和排列树。
针对这两个问题,算法的框架基本如下:
用回溯法搜索子集合树的一般框架:

Cpp代码
  1. void backtrack( int t){
  2. if (t>n)output(x);
  3. else {
  4. for ( int i=f(n,t);i<=g(n,t);i++){
  5. x[t]=h(i);
  6. if (constraint(t)&&bound(t))backtrack(t+1);
  7. }
  8. }
  9. }


用回溯法搜索排列树的算法框架:

Cpp代码
  1. void backtrack( int t){
  2. if (t>n)output(x);
  3. else {
  4. for ( int i=f(n,t);i<=g(n,t);i++){
  5. swap(x[t],x[i]);
  6. if (constraint(t)&&bound(t))backtrack(t+1);
  7. swap(x[t],x[i]);
  8. }
  9. }
  10. }


其中f(n,t),g(n,t)表示当前扩展结点处未搜索过的子树的起始标号和终止标号,
h(i)表示当前扩展节点处,x[t]第i个可选值。constraint(t)和bound(t)是当前
扩展结点处的约束函数和限界函数。constraint(t)返回true时,在当前扩展结点
x[1:t]取值满足约束条件,否则不满足约束条件,可减去相应的子树。bound(t)返
回的值为true时,在当前扩展结点x[1:x]处取值未使目标函数越界,还需要由backtrack(t+1)
对其相应的子树进一步搜索。
用回溯法其实质上是提供了搜索解空间的方法,当我们能够搜遍解空间时,
显然我们就能够找到最优的或者满足条件的解。这便是可行性的问题, 而效率可以
通过剪枝函数来降低。但事实上一旦解空间的结构确定了,很大程度上时间复杂度
也就确定了,所以选择易于搜索的解空间很重要。
下面我们看看两个最简单的回溯问题,他们也代表了两种搜索类型的问题:子集合问题和
排列问题。
第一个问题:
求集合s的所有子集(不包括空集),我们可以按照第一个框架来写代码:

Cpp代码
  1. #include<iostream>
  2. using namespace std;
  3. int s[3]={1,3,6};
  4. int x[3];
  5. int N=3;
  6. void print(){
  7. for ( int j=0;j<N;j++)
  8. if (x[j]==1)
  9. cout<<s[j]<< "" ;
  10. cout<<endl;
  11. }
  12. void subset( int i){
  13. if (i>=N){
  14. print();
  15. return ;
  16. }
  17. x[i]=1; //搜索右子树
  18. subset(i+1);
  19. x[i]=0; //搜索左子树
  20. subset(i+1);
  21. }
  22. int main(){
  23. subset(0);
  24. return 0;
  25. }



下面我们看第二个问题:排列的问题,求一个集合元素的全排列。
我们可以按照第二个框架写出代码:

Cpp代码
  1. #include<iostream>
  2. using namespace std;
  3. int a[4]={1,2,3,4};
  4. const int N=4;
  5. void print(){
  6. for ( int i=0;i<N;i++)
  7. cout<<a[i]<< "" ;
  8. cout<<endl;
  9. }
  10. void swap( int *a, int i, int j){
  11. int temp;
  12. temp=a[i];
  13. a[i]=a[j];
  14. a[j]=temp;
  15. }
  16. void backtrack( int i){
  17. if (i>=N){
  18. print();
  19. }
  20. for ( int j=i;j<N;j++){
  21. swap(a,i,j);
  22. backtrack(i+1);
  23. swap(a,i,j);
  24. }
  25. }
  26. int main(){
  27. backtrack(0);
  28. return 0;
  29. }


这两个问题很有代表性,事实上有许多问题都是从这两个问题演变而来的。第一个问题,它穷举了所有问题的子集,这是所有第一种类型的基础,第二个问题,它给出了穷举所有排列的方法,这是所有的第二种类型的问题的基础。理解这两个问题,是回溯算法的基础.
下面看看一个较简单的问题:
整数集合s和一个整数sum,求集合s的所有子集su,使得su的元素之和为sum。
这个问题很显然是个子集合问题,我们很容易就可以把第一段代码修改成这个问题的代码:

Cpp代码
  1. int sum=10;
  2. int r=0;
  3. int s[5]={1,3,6,4,2};
  4. int x[5];
  5. int N=5;
  6. void print(){
  7. for ( int j=0;j<N;j++)
  8. if (x[j]==1)
  9. cout<<s[j]<< "" ;
  10. cout<<endl;
  11. }
  12. void sumSet( int i){
  13. if (i>=N){
  14. if (sum==r)print();
  15. return ;
  16. }
  17. if (r<sum){ //搜索右子树
  18. r+=s[i];
  19. x[i]=1;
  20. sumSet(i+1);
  21. r-=s[i];
  22. }
  23. x[i]=0; //搜索左子树
  24. sumSet(i+1);
  25. }
  26. int main(){
  27. sumSet(0);
  28. return 0;
  29. }

回溯法之一---算法框架及基础


更多文章、技术交流、商务合作、联系博主

微信扫码或搜索:z360901061

微信扫一扫加我为好友

QQ号联系: 360901061

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描下面二维码支持博主2元、5元、10元、20元等您想捐的金额吧,狠狠点击下面给点支持吧,站长非常感激您!手机微信长按不能支付解决办法:请将微信支付二维码保存到相册,切换到微信,然后点击微信右上角扫一扫功能,选择支付二维码完成支付。

【本文对您有帮助就好】

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描上面二维码支持博主2元、5元、10元、自定义金额等您想捐的金额吧,站长会非常 感谢您的哦!!!

发表我的评论
最新评论 总共0条评论