Bootstrap

基础算法之搜素(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组成,每行每列,每小块不能有重复的数字出现。
请问:一共有多少种合法的填写方案?
在这里插入图片描述
具体解析,见洛谷【深基.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!

#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;
}
;