DFS(深度优先搜索)8种题型

 如果当时2020(不曾遗忘的符号) - 许嵩/朱婷婷 - 单曲 - 网易云音乐

 一个深爱的女孩 - 本兮 - 单曲 - 网易云音乐

 不说再见 - 好妹妹 - 单曲 - 网易云音乐

半年前写了一半的博客.......(2023/7/14),今天花6小时给它补充完毕......

目录

补充知识 

分类 

一, 1158: 八皇后

二, 3472. 八皇后

三,P1605 迷宫

四,P1434 [SHOI2002] 滑雪

AC  记忆化dfs

五,指数型枚举

六,排列型枚举

七,组合型枚举

八,油田 Oil Deposits

九,P1019 [NOIP2000 提高组] 单词接龙

总结


补充知识 

精益求精

在开始之前,我想先对BFS和DFS作个简单的区分

dfs可以用递归来写,也可以用来写;bfs可以用数组模拟队列,也可以直接stl的队列

dfs很难找到最优解,只用来找到某个解,内存消耗较小;bfs则适用于找最优解,但不适用于深层次的搜索

dfs如果需要回溯,就需要取消标记;而bfs不需要取消标记

接下来谈谈最近dfs刷题的思考:

关于,为什么有些dfs需要在标记,递归后进行取消标记,而另一些题目则不需要取消标记呢?

eg1: 给定起点,走5步,求所有可能到达的点

eg2: 给定起点,不限制步数,求能否到达终点

题1限制了步数,所以经过某点的路径会影响结果,那么就需要多次经过同一个点(意味着需要回溯),所以需要取消标记

问题2经过某点的路径不影响结果(不限制步数),所以不需要多次经过同一个点(不需要回溯),所以不需要取消标记

一次能把所有点走完,不需要取消标记(不需要回溯),走过的点就不会再走了

而如果部分路径需要多次访问,这时就需要回溯取消标记了,不然就没法访问了

-- -- End~

知识点 

下面,先结合百度,CSDN,博客园,知乎,《啊哈算法》,oi-wiki,增加对dfs的了解

(1条消息) 《啊哈算法》第四章之深度优先搜索_码龄?天的博客-CSDN博客

DFS(搜索) - OI Wiki (oi-wiki.org)

深度优先搜索 - AcWing

分类 

1,n皇后

基于国际象棋规则,皇后可以攻击(1)同一行(2)同一列,

(3)同一斜线(包括主对角线和副对角线的方向)  的其他单位

在n * n的棋盘中,放置n个皇后,使它们互相之间不攻击 

2,迷宫

迷宫里有几个点:

1,递归是很多人难以理解的,比如return;

2,模板则是,遍历 + 判断目标,其中遍历里的回溯 = 标记 + 递归 + 取消标记

3,走迷宫需要建立一个二维数组保存地图,并且需要一个二维方向数组

《啊哈算法》第四章之深度优先搜索_码龄?天的博客-CSDN博客

3,油田

有点类似记忆化dfs的过程,但是完全不一样。

记忆化dfs可以用一个数组保存过程,比如滑雪那题,找最长路程,后续遍历到以前走过的点,就不需要再遍历了,类似剪枝。

而油田,对每个点dfs后,将访问过的点在地图上直接操作(标记),省去了 vis[][] 标记数组,节省了空间。

4,单词接龙

5,记忆化dfs

6,dfs枚举

对dfs枚举做个总结,

1,都需要一个数组a[]来保存最后输出的答案

2,排列型枚举,就是全排列,dfs模板,可以画递归树辅助理解

3,组合型枚举,1~n中取m个数,由于是组合,打算顺序视作相同,如果处理呢?只需要保证后一个数 > 前一个数,加个 if 判断(还可以剪枝)

4,指数型枚举,和排列型稍作区别,排列型需要对每一个数进行选择,而指数型,只需要判断选 / 不选,不需要for(... i ~ n ...)

(1)指数型枚举

1~n中随机选择任意多个,输出所有方案的选择

int vis[N];
void dfs(int x)
{
    //到达边界
    if(x == n) { 
        ...
        return;
    }
    //只有选 / 不选两种可能
    //选
    vis[x] = 1; //标记选择
    dfs(x + 1); //递归
    vis[x] = 0; //取消标记
    //不选
    dfs(x + 1); //vis[x] = 0; 默认未选择
}

(2)组合型枚举

(3)排列型枚举(全排列)

一, 1158: 八皇后

P1158 - 八皇后 - New Online Judge (ecustacm.cn)

标签:基础题,USACO,深度优先搜索 

DFS(深度优先搜索)8种题型_第1张图片 DFS(深度优先搜索)8种题型_第2张图片DFS(深度优先搜索)8种题型_第3张图片下面展示两种做法

1,利用b, c, d三个数组,分别标记在某一列,某一右斜,某一左斜是否冲突

2,然后dfs(int row)的参数就是行数,避免了对第几行是否可行的讨论

3,在遍历的过程中递归

4,不是标准的dfs模板,但是做法值得学习,而且复杂度比直接套模板低

row就是行,j 就是列,row - j 就是主对角线,row + j 就是副对角线(row - j + 13防止负数

主对角线,row - j 相等;副对角线,row + j 相等 

DFS(深度优先搜索)8种题型_第4张图片

AC  代码

#include
int a[100], b[100], c[100], d[100];
int ans = 0, n;
using namespace std;
void dfs(int row)
{
    for(int j = 1; j <= n; ++j)
        if(b[j] == 0 && c[row - j + 13] == 0 && d[row + j] == 0) {
            a[row] = j;
            //标记
            b[j] = 1; //列已放置
            c[row - j + 13] = 1; //主对角线
            d[row + j] = 1; //副对角线
            if(row == n) { //得到一个答案
                ans++;
                if(ans <= 3) {
                    for(int k = 1; k <= n; ++k)
                        cout<>n;
    dfs(1);
    cout<

AC  代码2

标准的dfs模板

代码2耗时555ms,代码1耗时355ms

关于abs的解释

实际上还是对行,列,主,副对角线的判断,不过这次主副对角线一行搞定

dfs中for循环里,要放置的是第step + 1行, 第 j 列。前面遍历的是第 i 行第 a[i] 列
行的差值 == 列的差值   表示主副对角线有重合,就break

#include
using namespace std;
int a[100], vis[100], n, ans; //vis[j]统计第 j 列是否被放置

void dfs(int step) //已放置step个皇后, 第 step + 1行
{
    if(step == n) {
        ans++;
        if(ans <= 3) {
            for(int k = 1; k <= n; ++k)
                cout<>n;
    dfs(0); //第1行开始
    cout<

二, 3472. 八皇后

3472. 八皇后 - AcWing题库

标签:中等,DFS,北京大学考研机试题 

DFS(深度优先搜索)8种题型_第5张图片 DFS(深度优先搜索)8种题型_第6张图片在第一题的基础上,将结果存起来,不用排序!!!

主要目的是练手,再敲一遍,增加熟练度

卡点 

1,在dfs函数中,就可以用 *= 10的方法操作

2,最好不要用string b[]这样保存,否则还得加数字转化为字符串再 +=

AC  代码

#include
using namespace std;
int a[100], vis[100], n = 8, ans; //vis[j]统计第 j 列是否被放置
int b[100];

void dfs(int step) //已放置step个皇后, 第 step + 1行
{
    if(step == n) {
        ans++;
        for(int k = 1; k <= n; ++k) {
            b[ans] *= 10; //保存答案
            b[ans] += a[k];
        }
        return;
    }
    for(int j = 1; j <= n; ++j) { //第 j 列
        if(!vis[j]) { //第 j 列未放置
            int flag = 1;
            for(int i = 1; i <= step; ++i) //第 i 行
                if(abs(step + 1 - i) == abs(j - a[i])) {
                    flag = 0;
                    break;
                }
            if(flag) { //可以放置
                a[step + 1] = j;
                vis[j] = 1; //标记
                dfs(step + 1); //递归
                vis[j] = 0; //取消标记
            }
        }
    }
}

int main()
{
    dfs(0); //第1行开始
    int t, question;
    cin>>t;
    while(t--) {
        cin>>question;
        cout<
6
1
15863724
2
16837425
19
36271485
92
84136275
88
75316824
66
62713584
        

三,P1605 迷宫

P1605 迷宫 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

标签:DFS,USACO,枚举,普及-

DFS(深度优先搜索)8种题型_第7张图片

 DFS(深度优先搜索)8种题型_第8张图片

第一次,我忘记了标记初始点走过了。。还好在自己设计的样例里检测出来 

墙裂建议不会dfs的看看这个博客,里面的迷宫题模板,直接套在本题即可

(1条消息) 《啊哈算法》第四章之深度优先搜索_码龄?天的博客-CSDN博客

通过二维数组a[10][10]保存迷宫地图,障碍就a[i][j] = 1; 已走过的点就a[i][j] = -1,开始先初始化为a[i][j] = 0,表示都未走过 

AC  代码

如果next数组声明为全局变量,会报错 the reference to next is ambiguous,因为next是C++标准库中一个函数,会产生命名冲突,所以,声明全局的话,next[][]改成dir[][]即可 

#include
using namespace std;
int a[10][10], ans = 0;
int sx, sy, fx, fy, n, m;
void dfs(int x, int y)
{
    //到达目标
    if(x == fx && y == fy) {
        ans += 1;
        return; //回溯
    }
    int next[4][2] = { //方向数组, 循环得到下一步坐标
        {-1, 0}, //上
        {1, 0},  //下
        {0, -1}, //左
        {0, 1}}; //右
    //枚举四种走法
    int tx, ty; //临时变量
    for(int i = 0; i < 4; ++i) {
        tx = x + next[i][0];
        ty = y + next[i][1];
        //判断越界
        if(tx < 1 || ty < 1 || tx > n || ty > m)
            continue; //换个方向
        //不是陷阱且未走过
        if(a[tx][ty] == 0) {
            a[tx][ty] = -1; //标记走过
            dfs(tx, ty);   //递归
            a[tx][ty] = 0; //取消标记
        }
    }
}
int main()
{
    int t;
    cin>>n>>m>>t;
    cin>>sx>>sy>>fx>>fy;
    a[sx][sy] = -1; //标记初始走过
    int h, y;
    for(int i = 0; i < t; ++i) {
        cin>>h>>y;
        a[h][y] = 1; //1为障碍
    }
    dfs(sx, sy);
    cout<

四,P1434 [SHOI2002] 滑雪

P1434 [SHOI2002] 滑雪 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

标签:搜索,递归,记忆化搜索,普及/提高-

一样的题,同样的代码能AC

DFS(深度优先搜索)8种题型_第9张图片

DFS(深度优先搜索)8种题型_第10张图片

这是一道20年前的省选题

这里采用记忆化dfs,听起来,记忆化,很高大上,实际上本题只多了2行代码。。。。。

总结就是dfs模板 + 2行代码 

思路

第一次没用记忆化写了个void dfs,只AC了30%....

int dfs(int x, int y),然后声明全局二维数组Max[110][110]表示从该点出发的最大距离

由样例可知,初始点也算1距离,所以最后ans + 1

至于记忆化的2行呢,请看下面代码

int Max[110][110];
int dfs(int x, int y)
{
    if(Max[x][y]) return Max[x][y];
}

return Max[x][y],避免了重复计算,是不是有点像dp(动态规划)

题解

P1434 [SHOI2002] 滑雪 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

AC  记忆化dfs

当然卡了好一会,看了别人题解才AC

解释下第20行,Max[x][y]表示(x, y)点出发的最大距离,Max[tx][ty]表示(tx, ty)出发的最大距离

由于(x, y) --> (tx, ty),所以后一个点距离+1,就等于前一个点可能的最大距离(利用前面已经保存的结果,就不用重新算一次已经算过的过程了)

#include
using namespace std;
int Max[110][110], e[110][110]; //Max数组保存该点滑行距离
int n, m;
int nex[4][2] = {{1,0},{-1,0},{0,1},{0,-1}}; //上下左右
//记忆化dfs类似DP
int dfs(int x, int y)
{
    //关键: 记忆化dfs
    if(Max[x][y]) return Max[x][y]; //已经得到的数据,就不用再算了
    int tx, ty;
    //遍历4个方向
    for(int i = 0; i < 4; ++i) {
        tx = x + nex[i][0];
        ty = y + nex[i][1];
        if(tx < 0 || ty < 0 || tx >= n || ty >= m)
            continue;
        if(e[x][y] > e[tx][ty]) {
            dfs(tx, ty);
            Max[x][y] = max(Max[x][y], Max[tx][ty] + 1); //这行也很关键
        }
    }
    return Max[x][y];
}
int main()
{
    cin>>n>>m;
    //读入地图
    for(int i = 0; i < n; ++i)
        for(int j = 0; j < m; ++j)
            cin>>e[i][j];
    //dfs遍历
    for(int i = 0; i < n; ++i)
        for(int j = 0; j < m; ++j)
            Max[i][j] = dfs(i, j);
    //遍历Max数组得到最大长度
    int num = 0;
    for(int i = 0; i < n; ++i)
        for(int j = 0; j < m; ++j)
            num = max(num, Max[i][j]);
    cout<

洛谷上,不用记忆化,直接dfs能AC  90%

因为不熟练,我又在蓝桥杯官网敲了一次

AC  代码2

滑行 - 蓝桥云课 (lanqiao.cn)

#include
using namespace std;
int Max[110][110], e[110][110];
int n, m, ans, nex[4][2] = {{1, 0},{-1,0},{0,1},{0,-1}};
int dfs(int x, int y)
{
  if(Max[x][y]) return Max[x][y];
  int tx, ty;
  for(int i = 0; i < 4; ++i) {
    tx = x + nex[i][0];
    ty = y + nex[i][1];
    if(tx < 0 || ty < 0 || tx >= n || ty >= m) 
      continue;
    if(e[x][y] > e[tx][ty]) {
      Max[x][y] = max(Max[x][y], dfs(tx, ty) + 1);
    }
  }
  return Max[x][y];
}
int main()
{
  cin>>n>>m;
  for(int i = 0; i < n; ++i)
    for(int j = 0; j < m; ++j)
      cin>>e[i][j];
  for(int i = 0; i < n; ++i)
    for(int j = 0; j < m; ++j)  
      Max[i][j] = dfs(i, j);
  for(int i = 0; i < n; ++i)
    for(int j = 0; j < m; ++j)
      ans = max(ans, Max[i][j]);
  cout<

五,指数型枚举

92. 递归实现指数型枚举 - AcWing题库

标签:递归,简单 

DFS(深度优先搜索)8种题型_第11张图片DFS(深度优先搜索)8种题型_第12张图片

1~n个数,随机选取任意多个,那么每个数只有选和不选2种可能

所以方案数 = 2^n,单个方案n个数,那么时间复杂度 = O(2^15 * 15) ≈ 7.5 * 10^5

暴力没问题

因为输出数据达10^5以上,考虑printf(),提高速度

AC  代码1

#include
#include //printf()
using namespace std;
const int N = 20;

int n, vis[N]; //vis表示该位置是否选了
void dfs(int x) //x表示当前位置
{
    if(x == n) { //到达边界
        for(int i = 0; i < n; ++i) //从前往后遍历每一个
            if(vis[i] == 1) //已选中
                printf("%d ", i + 1);
        printf("\n");
        return; //不要忘记return
    }
    //选
    vis[x] = 1; //标记选它
    dfs(x + 1); //递归
    vis[x] = 0; //取消标记
    //不选
    dfs(x + 1); //不选默认vis[x] == 0, 不需要处理
}

int main()
{
    cin>>n;
    dfs(0);

    return 0;
}
4
1 2 3 4
1 2 3
1 2 4
1 2
1 3 4
1 3
1 4
1
2 3 4
2 3
2 4
2
3 4
3
4

 

补充知识点

1,

2的多少次方

“<<”是按位左移运算符,它会将某个二进制数向左移动指定的位数。例如,1 << 15表示将二进制数1向左移动15位,即:

0000000000000001 << 15 = 1000000000000000

所以,1 << 15在C++中表示的是二进制数1000000000000000

这个表达式可以用来快速计算2^15的值

具体来说,对于任意一个非负整数n,2^n可以表示为1左移n位,即1 << n 

2,

puts("")相当于回车,puts相当于输出一个字符串 + 回车,当字符串为空,等价于只输出回车

AC  代码2

我们也可以把两个printf删掉,将所有方案用一个二维数组ways统计

#include
#include //printf()
using namespace std;
const int N = 20;

int ways[1 << 15][16], cnt; //第cnt种方案
int n, vis[N]; //vis表示该位置是否选了
void dfs(int x) //x表示当前位置
{
    if(x == n) { //到达边界
        for(int i = 0; i < n; ++i) //从前往后遍历每一个
            if(vis[i] == 1) //已选中
                ways[cnt][i] = i + 1; //下标0开始,要加1
        cnt++; //方案书 +1
        return; //不要忘记return
    }
    //选
    vis[x] = 1; //标记选它
    dfs(x + 1); //递归
    vis[x] = 0; //取消标记
    //不选
    dfs(x + 1); //不选默认vis[x] == 0, 不需要处理
}

int main()
{
    cin>>n;
    dfs(0);

    //输出方案数
    for(int i = 0; i < cnt; ++i) {
        for(int j = 0; j < n; ++j) {
            if(ways[i][j]) printf("%d ", ways[i][j]);
        }
        puts(""); //换行
    }

    return 0;
}

六,排列型枚举

复杂度O(n * n!)

94. 递归实现排列型枚举 - AcWing题库

标签:简单,递归

DFS(深度优先搜索)8种题型_第13张图片

递归树辅助理解

DFS(深度优先搜索)8种题型_第14张图片

我们采取枚举位置的方法,枚举位置x,再用for循环判断放第i个数

AC  代码

#include
#include //printf()
using namespace std;
const int N = 10;

int n, vis[10], a[N];
void dfs(int x)
{
    if(x == n + 1) { //边界
        for(int i = 1; i <= n; ++i)
            printf("%d ", a[i]);
        puts(""); //换行
    }
    for(int i = 1; i <= n; ++i)
        if(vis[i] == 0) { //数字i未选择
            vis[i] = 1; //标记i已选择
            a[x] = i; //数字i放入第x个坑
            dfs(x + 1); //递归下一个位置
            //恢复现场
            //a[x] = 0; //这步不需要, 每一次开始前都会被覆盖
            vis[i] = 0; //取消数字i的标记
        }
}

int main()
{
    cin>>n;
    dfs(1); //第1个位置开始

    return 0;
}
4
1 2 3 4
1 2 4 3
1 3 2 4
1 3 4 2
1 4 2 3
1 4 3 2
2 1 3 4
2 1 4 3
2 3 1 4
2 3 4 1
2 4 1 3
2 4 3 1
3 1 2 4
3 1 4 2
3 2 1 4
3 2 4 1
3 4 1 2
3 4 2 1
4 1 2 3
4 1 3 2
4 2 1 3
4 2 3 1
4 3 1 2
4 3 2 1
 

七,组合型枚举

93. 递归实现组合型枚举 - AcWing题库

组合型,只是比排列型,多了个限制条件,由于不能重复,也就是123 231 132等等默认是一种情况,我们可以要求后一个比前一个大

只需在for循环的 if 中加个判断

注意!代码第15 ~16行 if 的剪枝,没有也可以,但是会慢很多(100ms和1162ms的区别),剪枝,意思是不符合条件时终止后面的运算(具体时间差多少请看)

DFS(深度优先搜索)8种题型_第15张图片

AC  代码

#include
using namespace std;
const int maxn = 26;

int n, m, a[26];

void dfs(int step) //已选step个数字
{
    if(step == m) {
        for(int i = 0; i < m; ++i)
            cout<>n>>m;
    dfs(0);

    return 0;
}

八,油田 Oil Deposits

标签:普及/提高-,DFS,连通块

油田 Oil Deposits - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

void dfs()中,递归dfs(tx, ty);前,要加个判断 if(a[tx][ty] == '@'),否则,不是油田是不能走的,疏忽了

AC  代码

UVA好像崩了,复制其他题解提交也是Unkonwn error,应该是AC了

#include
#include //memset()
using namespace std;
int n, m;
char a[110][110];

void dfs(int x, int y)
{
    a[x][y] = '*'; //标记
    int dir[8][2] = {
        {1,0},{-1,0},{0,1},{0,-1},{-1,1},{-1,-1},{1,1},{1,-1}
    }; //8个方向
    int tx, ty;
    for(int i = 0; i < 8; ++i) {
        tx = x + dir[i][0], ty = y + dir[i][1];
        if(tx < 0 || ty < 0 || tx >= n || ty >= m)
            continue;
        if(a[tx][ty] == '@') //只有油田才能走
            dfs(tx, ty);
        //连通块不需要取消标记
    }
}

int main()
{
    while(cin>>n>>m) {
        if(!n && !m) break;
        int ans = 0;
        for(int i = 0; i < n; ++i)
            for(int j = 0; j < m; ++j)
                cin>>a[i][j]; //读入油田
        for(int i = 0; i < n; ++i)
            for(int j = 0; j < m; ++j)
                if(a[i][j] == '@') { //从一个点出发
                    ans++;
                    dfs(i, j); //遍历 -- 所有能到达的'@'标记为'*'
                }
        cout<

九,P1019 [NOIP2000 提高组] 单词接龙

标签:字符串,搜索,普及/提高-

P1019 [NOIP2000 提高组] 单词接龙 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

一道上古时代的题....(要注意的点挺多的,都在注释里)

解释1

具体解释下,main函数中,初始dfs()的传参

dfs(" "+s[n], 1);

一开始,关于如何处理,用特定字符开头,挺苦恼的,所以采取这种方式传参

用样例举例

" "空格  +  字符串"a",当遇到 at,考虑到Link()函数中的dfs()递归

dfs(s[i], len_now + s[i].size() - k); //递归

递归后的长度(即a 和 at 拼凑后的长度),就是 1 + 2 - 1 == 2,也就实现了给定字符开头

避免了开头的 a 因为被包含而被处理掉

解释2

题目原文“另外相邻的两部分不能存在包含关系,例如 at 和 atide 间不能相连。”

这里其实有歧义,不是不能包含,而是包含后长度必须增加,这里只可意会

AC  代码

#include
using namespace std;

string s[30];
int use[30], ans, n; //use[]统计使用次数

int Link(string s1, string s2) //返回s1, s2最小重叠长度
{
    int Min_len = min(s1.size(), s2.size());
    for(int i = 1; i < Min_len; ++i) {//i 重叠长度
        int flag = 1;
        for(int j = 0; j < i; ++j)  //j s2中下标
            if(s1[s1.size() - i + j] != s2[j]) { //不匹配
                flag = 0;
                break;
            }
        if(flag && i != Min_len) return i; //i!=Min_len 不存在包含
    }
    return 0; //最小重叠长度为0
}

void dfs(string now, int len_now) //len_now 拼凑后的长度
{
    ans = max(ans, len_now); //更新答案
    for(int i = 0; i < n; ++i) {
        if(use[i] >= 2) continue; //已选2次, 不能再选了
        int k = Link(now, s[i]); //后接s[i]的重叠长度
        if(k) { //满足递归条件
            use[i]++; //标记
            dfs(s[i], len_now + s[i].size() - k); //递归
            use[i]--; //取消标记
        }
    }
}

int main()
{
    cin>>n;
    for(int i = 0; i <= n; ++i)
        cin>>s[i]; //读入字符串, s[n]为起始字符串
    dfs(" "+s[n], 1);
    cout<

总结

(一)n皇后,横 + 纵 + 主副对角线,用数组标记或找规律break
(二)迷宫,方向数组 + dfs模板,可在二维数组直接标记,注意传参
(三)滑雪,记忆化 + dfs模板,记忆化通过Max[][]避免重复搜索
(四)指数型枚举,1~n取任意个数,只有选 / 不选2种,将dfs模板for循环的遍历变成对2种情况的选择
(五)排列型枚举,全排列,dfs模板,画递归树辅助理解
(六)组合型枚举,1~n取m个数,dfs模板 + 限制条件(新取的数字要比上一个大),可剪枝(未选个数 < 待选个数 -- break)
(七)油田,连通块dfs,8个方向,对每个点dfs,将走过的点标记
(八)单词接龙,(1)初始传参,dfs(" "+"...", 1)对初始字母开头进行处理(2)Link()返回重叠长度(3)包含后长度一样的处理...

你可能感兴趣的:(2024蓝桥杯备赛,深度优先,深度优先遍历,蓝桥杯,算法)