DFS(小白式超详细讲解以及代码讲解)

2023-11-03 01:10

本文主要是介绍DFS(小白式超详细讲解以及代码讲解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

图的遍历算法是求解图的连通性,拓扑排序和关键路径等算法的基础。

根剧搜索路径的方向,通常有两条遍历图的路径:
深度优先搜索(DFS)和广度优先搜索(BFS)。
对于有向图和无向图都适用。

DFS
DFS类似于树的先序遍历,是树的先序遍历的推广。
那么对于一个联通图来说,深度搜索遍历的过程如下:
1.从图中某个顶点v出发,访问v;
2.找出刚访问过得顶点的第一个未被访问的邻接点,访问该顶点。以该顶点为新顶点,重复此步骤,直至访问过的顶点没有未被访问的邻接点为止。
3.返回前一个访问过的且乃有未被访问的邻接点的顶点,找出下一个未被访问的邻接点,访问该顶点。
4.重复(2)(3),直至所以的顶点都被访问过,搜索结束。
在这里插入图片描述
深度优先搜索遍历连通图
1)从图中某个顶点出发,访问v,并置visited[v]的值为true.
2) 依次检查v的所有的邻接点w,如果visited[w]的值为false,再从w出发进行递归遍历,直至图中所有的顶点都被访问过。

代码实现如下:

bool visited[maxn];//访问标志数组,初值为false;void DFS(Graph G,int v){//从顶点v出发递归的深度优先遍历图Gcout<<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个顶点出发深度优先搜索遍历图Gcout<<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未被访问,则递归调用DFSDFS(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;elsereturn 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;
}

我简单的写了一下运行过程,大家可以自己调试着这段程序理解一下!
在这里插入图片描述

这篇关于DFS(小白式超详细讲解以及代码讲解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/334534

相关文章

Python中Tkinter GUI编程详细教程

《Python中TkinterGUI编程详细教程》Tkinter作为Python编程语言中构建GUI的一个重要组件,其教程对于任何希望将Python应用到实际编程中的开发者来说都是宝贵的资源,这篇文... 目录前言1. Tkinter 简介2. 第一个 Tkinter 程序3. 窗口和基础组件3.1 创建窗

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

Java 接口定义变量的示例代码

《Java接口定义变量的示例代码》文章介绍了Java接口中的变量和方法,接口中的变量必须是publicstaticfinal的,用于定义常量,而方法默认是publicabstract的,必须由实现类... 在 Java 中,接口是一种抽象类型,用于定义类必须实现的方法。接口可以包含常量和方法,但不能包含实例

使用Redis实现会话管理的示例代码

《使用Redis实现会话管理的示例代码》文章介绍了如何使用Redis实现会话管理,包括会话的创建、读取、更新和删除操作,通过设置会话超时时间并重置,可以确保会话在用户持续活动期间不会过期,此外,展示了... 目录1. 会话管理的基本概念2. 使用Redis实现会话管理2.1 引入依赖2.2 会话管理基本操作

mybatis-plus分表实现案例(附示例代码)

《mybatis-plus分表实现案例(附示例代码)》MyBatis-Plus是一个MyBatis的增强工具,在MyBatis的基础上只做增强不做改变,为简化开发、提高效率而生,:本文主要介绍my... 目录文档说明数据库水平分表思路1. 为什么要水平分表2. 核心设计要点3.基于数据库水平分表注意事项示例

Nginx服务器部署详细代码实例

《Nginx服务器部署详细代码实例》Nginx是一个高性能的HTTP和反向代理web服务器,同时也提供了IMAP/POP3/SMTP服务,:本文主要介绍Nginx服务器部署的相关资料,文中通过代码... 目录Nginx 服务器SSL/TLS 配置动态脚本反向代理总结Nginx 服务器Nginx是一个‌高性

HTML5的input标签的`type`属性值详解和代码示例

《HTML5的input标签的`type`属性值详解和代码示例》HTML5的`input`标签提供了多种`type`属性值,用于创建不同类型的输入控件,满足用户输入的多样化需求,从文本输入、密码输入、... 目录一、引言二、文本类输入类型2.1 text2.2 password2.3 textarea(严格

JAVA项目swing转javafx语法规则以及示例代码

《JAVA项目swing转javafx语法规则以及示例代码》:本文主要介绍JAVA项目swing转javafx语法规则以及示例代码的相关资料,文中详细讲解了主类继承、窗口创建、布局管理、控件替换、... 目录最常用的“一行换一行”速查表(直接全局替换)实际转换示例(JFramejs → JavaFX)迁移建

Go异常处理、泛型和文件操作实例代码

《Go异常处理、泛型和文件操作实例代码》Go语言的异常处理机制与传统的面向对象语言(如Java、C#)所使用的try-catch结构有所不同,它采用了自己独特的设计理念和方法,:本文主要介绍Go异... 目录一:异常处理常见的异常处理向上抛中断程序恢复程序二:泛型泛型函数泛型结构体泛型切片泛型 map三:文

SQL Server中行转列方法详细讲解

《SQLServer中行转列方法详细讲解》SQL行转列、列转行可以帮助我们更方便地处理数据,生成需要的报表和结果集,:本文主要介绍SQLServer中行转列方法的相关资料,需要的朋友可以参考下... 目录前言一、为什么需要行转列二、行转列的基本概念三、使用PIVOT运算符进行行转列1.创建示例数据表并插入数