基础算法之搜素(bfs和dfs模板和例题)

这篇具有很好参考价值的文章主要介绍了基础算法之搜素(bfs和dfs模板和例题)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


之前学习了暴力枚举策略,将所有可能的情况都枚举一遍以获得最优解,但是枚举全部元素的效率如同愚公移山,无法应付数据范围稍大的情形。
本节在暴力枚举的基础上介绍搜索算法,包括 深度优先搜索和广度优先搜索从起点开始,逐渐扩大寻找范围,直到找到需要的答案为止
严格来说,搜索算法也算是种暴力枚举策略,但是其算法特性决定了效率比直接的枚举所有答案要高,因为搜索可以跳过一些无效状态,降低问题规模。在算法竞赛中,如果选手无法找到一种高效求解的方法(如贪心递推、动态规划公式推导等,使用搜索也可以解决一些规模较小的情况;而有的任务就是必须使用搜索来完成,因此这是相当重要的策略。

一、深度优先搜索与回溯

DFS模板(回溯):
dfs和回溯的关系(没用,可以不看)
主要用于解决:排列问题,组合问题,切割问题,棋盘问题等

void dfs(int k){  
//k代表递归层数,也就是填写到第几个空了,或者搜索到第几个空 
	if(k>=n){//空已经填完了 
		判断最优解/记录答案/输出符合要求的排列 
		return; 
	} 
	
	for(int i = 0; i < ;i++){
	 //枚举提供给这个空填写的选项;迷宫类问题是枚举四个方位 
		if(jugde(i)){ //判断这个选项是合法的,如:判断边界;特殊条件判断
			记录这个空填的数以及对应的状态值修改 
			dfs(k+1)//  继续判断下一层
			取消对当前这个空的处理 恢复现场 
		}
	}
} 

1、四阶数独

数独是一种著名的益智游戏,这里讨论的是一种简化后的数独-四阶数独,给出一个4*4的方格,每个格子只能填写1-4的整数,要求每行每列和四等分更小的正方形部分刚好都由1-4组成,每行每列,每小块不能有重复的数字出现。
请问:一共有多少种合法的填写方案?
基础算法之搜素(bfs和dfs模板和例题),算法学习和刷题(acm、蓝桥杯、cf),算法,宽度优先,深度优先
具体解析,见洛谷【深基.14 搜索】p188
这个博主讲得也还可以:链接

解题代码:

#include<bits/stdc++.h>
using namespace std;
int b1[5][5],b2[5][5],b3[5][5]; 分别标记每行、每列、每个小块出现的数字 
int cnt=0; 
int g[20];    //记录棋盘 
void dfs(int x){
	if(x>16){  //如果取够数 
		cnt++;  
		/*打印所有方案 
		for(int i = 1; i <= 16;i++){
				cout<<g[i]<<" ";
			if(i%4==0) cout<<endl;
		}
		cout<<endl;
		*/
	}
	int row = (x-1)/4 + 1;  //找到这个空在第几行 
	int col = (x-1)%4 + 1;  //得到这个空在第几列 
	int exd;//一共就四个块 可以特判 也可以直接用规律
	exd = (row-1)/2*2 + (col-1)/2 +1;// 根据行列值,四个小块可以表示为0++0+1 0+1+1 2+0+1 2+1+1四种情况 
	for(int i = 1; i <= 4; i++){
		//cout<<k<<" "<<i<<endl; 
		if(b1[row][i]==0&&b2[col][i]==0&&b3[exd][i]==0){  //判断是否符合条件 
			g[x] = i; //保存结果 
			b1[row][i]=1; b2[col][i]=1; b3[exd][i]=1;//保存状态(占位) 
			//cout<<g[k]<<endl;
			dfs(x+1);//下一层递归 
			b1[row][i]=0; b2[col][i]=0; b3[exd][i]=0; //恢复状态 (取消占位) 
		} 
	} 
}
//memset(b1,0,sizeof(b1));
int main(){
	dfs(1); 
	cout<<cnt;
	return 0;
} 

2、排列类问题

遇到了需要枚举排序的时候,搜索回溯会很好用。不需要生成所有的序列全排列,而是一一个一个地填空,保证填空的时候序列是合法的,这样就可以不用枚举很多无效序列,节约程序运行时间。

枚举的三大类型:博客链接
1、递归实现指数型枚举:
主要题干:从 1∼n 这 n 个整数中随机选取任意多个,输出所有可能的选择方案。 同一行内的数必须升序排列,相邻两个数用恰好 1 个空格隔开。对于没有选任何数的方案,输出空行。本题有自定义校验器(SPJ),各行(不同方案)之间的顺序任意。状态空间规模:2n
思路: 递归搜索树,对每一个数字,有选和不选它,两种状态,放入到答案数组里面。递归选和不选两个子解空间就行了

#include<bits/stdc++.h>
using namespace std;
vector<int> ans;
int n;
void solve(int x){
	if(x>n){
		for(int i = 0; i < ans.size(); i++){
			printf("%d ",ans[i]);
		}cout<<'\n';
		return;
	}
	
	solve(x+1);   //这个位置的数,没被选择; 进入下一个空 
	ans.push_back(x);
	solve(x+1);   //这个位置的数,被选择; 进入下一层 
	ans.pop_back(); //回到上一层时,这个数是没有杯加入到答案数列里面的 
} 
int main(){
	cin>>n;
	solve(1);	
	return 0;
}

2、递归实现组合型枚举:
主要题干:从 1∼n 这 n 个整数中随机选出 m 个,输出所有可能的选择方案。 首先,同一行内的数升序排列,相邻两个数用一个空格隔开。其次,对于两个不同的行,对应下标的数一一比较,字典序较小的排在前面(例如 1 3 5 7 排在 1 3 6 8 前面)。
状态空间规模:C(nm)
思路:在指数型枚举的基础上,进行剪枝,本题中, 如果已经选择了超过m个数,或者即使再选上剩余所有的数也不够m个,就可以提前返回了。 因为无解。

#include<bits/stdc++.h>
using namespace std;
vector<int> ans;
int n,m;
void solve(int x){
	if(ans.size()>m||(ans.size()+n-x+1)<m)  return;
	if(x>n){
		for(int i = 0; i < m; i++){
			printf("%d ",ans[i]);
		}cout<<'\n';
		return;
	}
	//先进行“选”的分支,这样字典序小的才能在前面
	ans.push_back(x);
	solve(x+1);   //这个位置的数,被选择; 进入下一层 
	ans.pop_back(); //回到上一层时,这个数是没有杯加入到答案数列里面的 
	
	solve(x+1);   //这个位置的数,没被选择; 进入下一个空 
} 
int main(){
	cin>>n>>m;
	solve(1);	
	return 0;
}

3、递归实现排列型枚举(全排列)
把 1∼n 这 n 个整数排成一行后随机打乱顺序,输出所有可能的次序。
状态空间规模:n!

1、2 与3 的做法是有一些不一样的,全排列的做法是规规矩矩的dfs模板做法。

#include<bits/stdc++.h>
using namespace std;
int n;
int ans[550],stu[550];
void dfs(int x){
	if(x>n){
		for(int i = 1; i< n+1;i++){
			printf("%d ",ans[i]);
		}cout<<'\n';
	}
	for(int i = 1; i <= n; i++){
		if(stu[i]==0){
			stu[i]=1; ans[x]= i;
			dfs(x+1);
			stu[i] = 0; ans[x] = 0; 
		}
	}
}
int main(){
	cin>>n;
	dfs(1);
	return 0;
}

Dfs求解有重复元素的全排列
重点是可填写数据的遍历和符合条件判断。
https://blog.csdn.net/qq_52626583/article/details/123572390?spm=1001.2014.3001.5502

#include<stdio.h>
char s[550]; //读入字母元素 
int num[27];
char p[550];  //输出字母元素 
int n;
int sum = 0;
void dfs(int x){
	if(x==n){  //填写长度为n的序列 
		sum++;
		printf("%s\n",p);
	}else{
		for(int i = 1; i <=26;i++){ //可用于填写的所有字母 
			if(num[i]){  //判断这个字母还有才能用于填写 
				p[x] = i-1 + 'a' ;
				num[i]--;
				dfs(x+1);
				num[i]++;
			}
		}
	}
}
int main(){
	scanf("%d",&n);scanf("%s",s);
	for(int i = 0; i< n;i++){  
		num[s[i]-'a'+1] ++;
	}	
	dfs(0);
	printf("%d",sum);
	return 0;
} 

3、红与黑(dfs或bfs和Flood fill)

题目链接
有一间长方形的房子,地上铺了红色、黑色两种颜色的正方形瓷砖。你站在其中一块黑色的瓷砖上,只能向相邻(上下左右四个方向)的黑色瓷砖移动。请写一个程序,计算你总共能够到达多少块黑色的瓷砖。
输入格式:
输入包括多个数据集合。
每个数据集合的第一行是两个整数 W 和 H,分别表示 x 方向和 y 方向瓷砖的数量。
在接下来的 H 行中,每行包括 W 个字符。每个字符表示一块瓷砖的颜色,规则如下
1)‘.’:黑色的瓷砖;
2)‘#’:红色的瓷砖;
3)‘@’:黑色的瓷砖,并且你站在这块瓷砖上。该字符在每个数据集合中唯一出现一次。
当在一行中读入的是两个零时,表示输入结束。
输出格式:
对每个数据集合,分别输出一行,显示你从初始位置出发能到达的瓷砖数(记数时包括初始位置的瓷砖)。
数据范围:
1≤W,H≤20
输入样例:

6 9 
....#. 
.....# 
...... 
...... 
...... 
...... 
...... 
#@...# 
.#..#. 
0 0

输出样例:

45

解题思路:

AC代码: bfs!

#include<iostream>
#include<algorithm>
#include<queue>
#include<cstring>
using namespace std;
const int N = 25;
char g[N][N];
int n,m;
int dx[5] = {-1,0,1,0},dy[5] = {0,1,0,-1};
struct node{
    int x,y;    
}tmp;
int bfs(int sx,int sy){
    int res = 0;
	queue<node> q;
    q.push({sx,sy});
    g[sx][sy] = '#';
    while(q.size()){
    	tmp = q.front();
    	q.pop();
    	res++;
    	for(int i = 0; i <4;i++){
    		int x = tmp.x + dx[i],y = tmp.y + dy[i];
    		if(x<0||x>=n||y<0||y>=m||g[x][y]=='#') continue;
			q.push({x,y});
			g[x][y] = '#';
		}
	}
	return res;
}
int main(){
	int ans = 0;
	
	while(cin>>m>>n,m||n){
		for(int i = 0; i < n;i++) cin>>g[i];
    	
    	for(int i = 0;i <n;i++){
        	for(int j = 0; j <m;j++){
            	if(g[i][j]=='@')  ans = bfs(i,j);
        	}
		}
		
		cout<<ans<<endl;
	}
    
    return 0;
}

AC代码 dfs!文章来源地址https://www.toymoban.com/news/detail-565649.html

#include <cstring>
#include <iostream>
using namespace std;

const int N = 25;

int n, m;
char g[N][N];//存储地板

int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};//上右下左四个方向

int dfs(int x, int y)//深度优先遍历
{
    int cnt = 1;

    g[x][y] = '#';
    for (int i = 0; i < 4; i ++ )//走四个方向
    {
        int a = x + dx[i], b = y + dy[i];
        if (a < 0 || a >= n || b < 0 || b >= m) continue;
        if (g[a][b] != '.') continue;
        cnt += dfs(a, b);//如果可以走向某一个方向,对该方向上的点递归
    }
    return cnt;
}

int main()
{
    int ans;
	
	while (cin >> m >> n, n || m){
        for (int i = 0; i < n; i ++ ) cin >> g[i];//一次读入一行

        for (int i = 0; i < n; i ++ )
            for (int j = 0; j < m; j ++ )
                if (g[i][j] == '@')	ans = dfs(i,j);
        
        cout << ans << endl;
    }
    
    return 0;
}

到了这里,关于基础算法之搜素(bfs和dfs模板和例题)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 红与黑(bfs + dfs 解法)(算法图论基础入门)

    献给阿尔吉侬的花束( 入门级bfs查找 + 模版解读 + 错误示范 在之前的博客当中,详细地介绍了这类题目的解法,今天为大家带来一道类似的题目练练手,后续还会更新更有挑战的题目以及更为详细的解析,喜欢的小伙伴可以点个关注啦! 有一间长方形的房子,地上铺了红色、

    2024年01月20日
    浏览(43)
  • Python 算法基础篇:深度优先搜索( DFS )和广度优先搜索( BFS )

    深度优先搜索( DFS )和广度优先搜索( BFS )是两种常用的图遍历算法,用于在图中搜索目标节点或遍历图的所有节点。本篇博客将介绍 DFS 和 BFS 算法的基本概念,并通过实例代码演示它们的应用。 😃😄 ❤️ ❤️ ❤️ 深度优先搜索( DFS )是一种用于遍历或搜索图或树

    2024年02月07日
    浏览(52)
  • 【算法基础:搜索与图论】3.2 树与图的dfs和bfs

    要学会建树、建图的通用方法。 dfs 和 bfs 的代码框架。 https://www.acwing.com/problem/content/848/ 在 dfs 的过程中,统计各个节点作为断点时的连通块最大值。 https://www.acwing.com/problem/content/849/ 看到最短距离就可以想到使用宽搜。 注意! :题目中说明了 a 和 b 表示存在一条从 a 走到

    2024年02月16日
    浏览(29)
  • 拓扑排序算法 -- dfs、bfs

    210. 课程表 II 该题用到「拓扑排序」的算法思想,关于拓扑排序,直观地说就是,让你把⼀幅图「拉平」,⽽且这个「拉平」的图⾥⾯,所有箭头⽅向都是⼀致的,⽐如上图所有箭头都是朝右的。 很显然,如果⼀幅有向图中存在环,是⽆法进⾏拓扑排序的,因为肯定做不到所

    2024年02月10日
    浏览(28)
  • 算法-图BFS/DFS-单词接龙

    https://leetcode-cn.com/problems/number-of-islands 给定两个单词(beginWord 和 endWord)和一个字典,找到从 beginWord 到 endWord 的最短转换序列的长度。转换需遵循如下规则: 每次转换只能改变一个字母。 转换过程中的中间单词必须是字典中的单词。 说明: 如果不存在这样的转换序列,返回

    2024年02月10日
    浏览(33)
  • 算法回忆录——DFS与BFS

    拿迷宫(二维,上下左右四个方向,入口在左方)来举例,一开始从入口出发,一直往右走,如果遇到了路口,就记录下该路口有几条路可以走,然后选择一条走下去,如果走下去又遇到了一个路口,那么又记录下这个路口有几条路口可以走,然后返回第一个路口,选择未走

    2024年01月21日
    浏览(29)
  • 算法设计与分析实验:DFS与BFS

    目录 一、边界着色 1.1 思路一:DFS 1.2 思路二:BFS 二、课程表II 2.1 思路一:DFS 2.2 思路二:拓扑排序 三、岛屿的最大面积 3.1 具体思路 3.2 思路展示 3.3 代码实现 3.4 复杂度分析 3.5 运行结果 2024-1-31 阴 力扣第1034题 (1)具体思路: 首先,定义一个dfs函数,用于搜索和染色连通

    2024年02月21日
    浏览(24)
  • BFS算法(宽度优先搜索)超强解析 BFS迷宫问题图文详解 DFS与BFS的区别

      前情回顾:DFS练习-迷宫(最短路径)问题详解 一波三折 图片+文字 以及你需要会的基础:手搓数据结构之队列queue C/C++语言版(BFS算法预备知识) 广度优先搜索(Breadth First Search)简称广搜或者 BFS, 是遍历图存储结构的一种算法 。 BFS的原理是 “逐层扩散” ,从起点出发

    2024年02月22日
    浏览(38)
  • 【算法】(BFS/DFS)迷宫路径问题(C语言)

    题目 :现有一迷宫如下图所示,蓝色部分为墙壁,白色部分为通路,入口在左上角(1,1)处,出口在右下角(8,8)处,试找出一条路径以通过该迷宫(路径不能重叠)。 分析 : ① 使用二维数组来存储迷宫,墙用 1 表示,路用 0 表示,如下图所示: 为与题目中的入口坐标

    2024年02月07日
    浏览(72)
  • DFS深搜算法(详解+例题)

    目录 Everyday English DFS是什么? DFS框架 例题详解 1、全排列问题(洛谷 P1706 C++)  题目概述 思路点拨 参考代码 2、自然数的拆分问题(洛谷 P2404 C++) 题目概述 ​编辑思路点拨 AC代码 结尾 Thank you for listening! Failure is the mother of success. 失败乃成功之母。 DFS是英文Depth-First-Sear

    2024年02月11日
    浏览(27)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包