DFS(小白式超详细讲解以及代码讲解)_dfs()这个代码怎么解释

bool visited[maxn];//访问标志数组,初值为false;

void DFS(Graph G,int v){//从顶点v出发递归的深度优先遍历图G
cout<<v;
visited[v] = true;
for(顶点v的第一个邻接顶点w;w >= 0;下一个邻接点)
if(!visited[w]) DFS(G,w);//对v尚未访问的邻接点w递归调用DFS;
}


那么对于非连通图的遍历,我们可以看做是一个个连通分量,循环调用多少次,那么就有多少个连通分量。  
 **用深度优先遍历非连通图**



void DFS(Graph G){//对非连通图G做深度优先遍历
for(v = 0;v < G.num;++v) visited[v] = false;
for(v = 0;v < G.num; ++v)//循环调用连通图遍历
if(!visited[v]) DFS(G,v);// 对未访问的顶点调用DFS;

}


我们知道,在调用DFS之前,我们需要选择合适的存储方式把我们的图存起来。


常见的存图方式有如下:


采用邻接矩阵表示图的深度优先搜索遍历



void DFS(Graph G,int v){//图G为邻接矩阵类型,从第v个顶点出发深度优先搜索遍历图G
cout<<v;
visited[v] = 1;
for(w = 0 ;w < G.num;w++)//依次检查邻接矩阵v所在的行
if((G.arcs[v][w] != 0)&&(!visted[w]))//G.arcs[v][w]表示w是v的邻接点,如果w未被访问,则递归调用DFS
DFS(G,w);
}


**采用邻接表表示的图深度优先搜索遍历**



void DFS(Graph G,int v){
cout<<v;
visited[v] = 1;
p = G.vertices[v].firstarc;//p指向v的边链表的第一个节点
while(p != NULL){//边链表非空

w = p -> adjvex;//如果w是v的邻接点
if(!visited[w]) DFS(G,w);//如果w未访问,则递归调用DFS;

p = p -> nextarc;//指向下一个边结点
}
}


好了,最基础的理论知识我们已经了解完了,接下来我们要跟深一步了解这个算法,并写代码做题了


DFS算法思想:一直往深处走,直到找到解或者走不下去为止;


一般DFS使用**栈**保存未被检测的结点,结点深度优先的次序被访问并被依次压入栈中,并以相反的次序出栈进行新的检测。


**深搜解决栗子:走迷宫。不撞南墙不回头!**


下面是我做题的一个基础模板!



#include<bits/stdc++.h>

using namespace std;

const int maxn = 100;

bool vis[maxn][maxn];//访问标记
int mapp[maxn][maxn];//坐标范围

bool check(int x,int y){//边界条件和约束条件的判断
if(!vis[x][y] && …)//满足条件
return 1;
else
return 0;
}
void DFS(int x,int y){
vis[x][y] = 1;//标记该节点被访问
if(mapp[x][y] == G){//出现目标态G
…//做相应处理
return ;
}
for(int i=0;i<4;i++){
if(check(x + dir[i][0],y+dir[i][1]))//按照规矩生成下一个节点
DFS(x + dir[i][0],y+dir[i][1]);
}
return ;//没有下层搜索节点,回溯
}
int main(){

return 0;
}


**DFS实现全排列**


思路:我们可以这样来想:  
 1.首先我们考虑1号盒子,我们约定每到一个盒子面前都按数字递增的顺序摆放扑克牌。于是把1号扑克牌放到1号盒子中。  
 2.接着考虑2号盒子,现在我们手里剩下2号和3号扑克牌,于是我们可以把2号扑克牌放入2号盒子中。于是在3号盒子只剩一种可能性,我们继续把3号扑克放入3号盒子。此时产生了一种排列——{1,2,3}。  
 3.接着我们收回3号盒子中的3号扑克牌,尝试一种新的可能,此时发现别无他选。于是选择回到2号盒子收回2号扑克。  
 4.在2号盒子中我们放入3号扑克,于是自然而然的在3号盒子中只能放入2号扑克。此时产生另一种排列——{1,3,2};  
 5.重复以上步骤就能得到数字{123}的全排列。



#include <bits/stdc++.h>
using namespace std;
int a[101],b[101],n;
void print()
{
int i;
for(i=1;i<=n;i++)
{
cout<<a[i]<<’ ';
}
cout<<endl;
}
inline void dfs(int i)
{
int j;
if(i==n+1)
{
print();
return;
}
for(j=1;j<=n;j++)
{
if(b[j]==0)
{
a[i]=j;
b[j]=1;
dfs(i+1);
b[j]=0;
}
}
}
int main()
{
ios::sync_with_stdio(false);
cin>>n;
dfs(1);
return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值