当前位置:   article > 正文

彻底搞懂dfs与回溯_dfs和回溯

dfs和回溯

目录

初识dfs 

 扩展到图


深度优先搜索dfs,其过程是对每一个可能的分支路径深入到不能再深入为止,是一种广泛用于树和图中搜索路径,和其他情况下搜索需要的情况的算法

初识dfs 

彻底弄懂dfs和回溯,我们先不讲图,先来个简单的

假如我们要把(1,2,3)这个集合所有的子集都搜索出来

也就是下面这个dfs(n等于3)

也就是下面这个代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. int n;
  4. vector<vector<int> > res;
  5. vector<int> temp;
  6. int tmp;
  7. void dfs(int idx){
  8. if(idx==n+1){
  9. res.push_back(temp);
  10. for(auto t:temp) cout<<t<<" ";
  11. cout<<endl;
  12. return ;
  13. }
  14. temp.push_back(idx);
  15. dfs(idx+1);
  16. temp.pop_back();
  17. dfs(idx+1);
  18. }
  19. signed main(){
  20. cin>>n;
  21. dfs(1);
  22. }

首先为什么子集能dfs出来呢,对于每一个数,都有要和不要两种答案

他们会产生两个不同的分支,进入下一层,而下一个数又有两种答案.....

比如不要1,要2,不要3,得到的子集就是(2),都不要得到的就是空集

我们画一颗搜索树便于我们理解

 一开始我们什么都没有,然后每一层都是要和不要,得到最后的所有答案

那么怎么用代码实现这东西呢

首先idx代表层,最开始等于1,每一层+1,当等于n+1时,说明已经全部走完了,此时temp中就是我们要的答案,这个很好理解

然后对于每一个idx,要这个数就是temp.push_back(idx),把这个数加进temp中,否则不要他直接dfs下一层,这里我们要回溯了,要把idx这个数pop掉,此时temp中没有这个数,才代表我们不要,然后dfs下一层

通俗的解释下回溯

回溯这个东西就是,假设你现在正处在上图的过程,有一个中间变量tmp的值为X1,这个中间变量可以是随便一个什么,int或string或数组或其他,而你通过过程去搜索可能1的时候,会改变这个中间变量tmp,假设你把他加上了d,现在中间变量tmp的值为tmp+d

而等你搜完可能1的时候,再去搜索可能2,可能2是要通过过程来的,也就是搜可能2的时候要求tmp的值为X1,但搜完可能1的时候,这个tmp的值为X1+d,所以在写代码的时候就要在 dfs可能1 的后面写上tmp要减去d,这样才能回复过程时的中间变量,才能继续搜索可能2,这就是回溯

简单来说,回溯就是要消除搜索过程中的不同可能性之间对中间变量的影响

我不知道我说明白没有,大家看下图 详细过程

如果听懂了

接下来试一个稍微深一点的例子

求长度为n,每个数字可以在1-m中选且不能重复的所有情况

就比如要求长为2,m等于3的所有情况

答案是(1 2)(1 3)(2 1)(2 3)(3 1)(3 2)

看看能不能理解这里的回溯

代码:

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. int n,m;
  4. vector<int> temp;
  5. int st[100];
  6. void dfs(int idx){
  7. if(idx==n+1){
  8. for(auto t:temp) cout<<t<<" ";
  9. cout<<endl;
  10. return ;
  11. }
  12. for(int i=1;i<=m;++i){
  13. if(st[i]==0){
  14. temp.push_back(i);
  15. st[i]=1;
  16. dfs(idx+1);
  17. st[i]=0;//回溯
  18. temp.pop_back();//回溯
  19. }
  20. }
  21. }
  22. signed main(){
  23. cin>>n>>m;
  24. dfs(1);
  25. }

这里st代表标记每个数用没用过,很明显,一种可能和另一种可能的st是不能互相干扰的

所以要消除每种可能之间的影响

 扩展到图

其实这就不难理解了

树本身就相当于一颗搜索树

而对于图,如果记录下每个点只遍历一次,就也变成了一颗搜索树

假设我们用dfs遍历这张图

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. vector<int> v[100];
  4. int n,m;
  5. int st[100];
  6. void dfs(int now){
  7. cout<<"遍历到 "<<now<<endl;
  8. for(auto t:v[now]){
  9. if(!st[t]){
  10. st[t]=1;
  11. dfs(t);
  12. }
  13. }
  14. }
  15. signed main(){
  16. cin>>n>>m;
  17. while(m--){
  18. int a,b;
  19. cin>>a>>b;
  20. v[a].push_back(b);
  21. v[b].push_back(a);
  22. }
  23. st[1]=1;
  24. dfs(1);
  25. }

输出如下

诶这对比之前那两个是不少了点啥

咋不回溯了呢

之前dfs后面不是得st[t]=0吗

对呀,咋不回溯了呢

大家可以写个回溯试试

 是不是重复遍历了很多点呀

因为遍历图的时候每个点只遍历一遍呀

如果回溯了,我现在这条路遍历到3这个点,之后消除影响了,那我其他另一条路在碰到3这个点是不是还要走一遍呀,那不就重复遍历了吗

这种情况是必须要不同可能之间要产生影响的,并且这个影响不能被消除,回溯不就是消除影响吗,所以不回溯了

如果以上都理解了,那dfs和回溯最底层的逻辑也就差不多这些了

睡觉睡觉..

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/478044
推荐阅读
相关标签
  

闽ICP备14008679号