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

news2024/12/25 9:17:16

目录

    • 一、深度优先搜索与回溯
      • 1、四阶数独
      • 2、排列类问题
      • 3、红与黑(dfs或bfs和Flood fill)

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

一、深度优先搜索与回溯

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/757661.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【LLM】Langchain使用[三](基于文档的问答)

文章目录 一、基于文档的问答1. 创建向量存储2. 不同类型的chain链 二、本地知识库问答Reference 一、基于文档的问答 1. 创建向量存储 使用Dock Array内存搜索向量存储&#xff0c;作为一个内存向量存储&#xff0c;不需要连接外部数据库创建向量存储&#xff1a;导入一个索…

复习第七课 C语言-指针数组,函数,string

目录 【1】指针和数组 【2】数组指针 【3】指针数组 【4】函数 【5】函数传参 【6】动态开辟堆区空间 【7】string函数族 【8】递归函数 练习&#xff1a; 【1】指针和数组 直接访问&#xff1a;通过数组名访问 间接访问&#xff1a;通过指针访问 》1. 一维数组 in…

动态规划之118杨辉三角(第6道)

题目&#xff1a;给定一个非负整数 numRows&#xff0c;生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中&#xff0c;每个数是它左上方和右上方的数的和。 题目链接&#xff1a;118. 杨辉三角 - 力扣&#xff08;LeetCode&#xff09; 示例&#xff1a; 解法&#xff1…

【iOS】内存管理五大区

参考博客&#xff1a;iOS内存管理学习第一篇-内存五大区 3.1 OC特性之 内存五大区域 1. 简述 程序要想执行&#xff0c;第一步就需要 被加载到内存中 内存五大区域: 栈区,堆区,BSS段(静态区),常量区(数据段),代码段. 栈区&#xff08;stack&#xff09;由编译器自动分配并释放…

手写Spring框架---MVC实现

目录 预备 自研框架MVC的实现 MVC架构草图&#xff1a; 大致流程 实现思路 自定义注解 JavaBean 请求的拦截-建立DispatcherServlet 责任链处理请求 RequestProcessor矩阵 Render矩阵 预备 在DispatcherServlet&#xff1a; 解析请求路径和请求方法依赖容器&#xf…

最全的 Spring 依赖注入方式,你都会了吗?

Spring 正如其名字&#xff0c;给开发者带来了春天&#xff0c;Spring 是为解决企业级应用开发的复杂性而设计的一款框架&#xff0c;其设计理念就是&#xff1a;简化开发。 Spring 框架中最核心思想就是&#xff1a; IOC&#xff08;控制反转&#xff09;&#xff1a; 即转移…

ChatGPT 最佳实践指南之:使用外部工具

Use external tools 使用外部工具 Compensate for the weaknesses of GPTs by feeding them the outputs of other tools. For example, a text retrieval system can tell GPTs about relevant documents. A code execution engine can help GPTs do math and run code. If a …

45、Spring Boot自动配置原理

Spring Boot自动配置原理 lmport Configuration Spring spi 自动配置类由各个starter提供&#xff0c;使用Configuration Bean定义配置类&#xff0c;放到META-INF/spring.factories下使用Spring spi扫描META-INF/spring.factories下的配置类使用lmport导入自动配置类

通讯录管理系统--进阶(动态开辟内存+保存数据到文件)

文章目录 动态开辟内存优化改进通讯录类型改进初始化通讯录函数改进添加联系人的函数增加销毁通讯录信息的函数 保存数据到文件优化保存通讯录数据到文件读取数据到通讯录 完整的代码展示 在 C语言实现通讯录的所有基本功能详细代码分析中&#xff0c;我们已经实现了通讯录的基…

Linux系统编程:文件系统和inode

目录 一. 磁盘的结构和读写数据的方式 1.1 磁盘级文件和内存级文件 1.2 磁盘的物理结构 1.3 访问磁盘数据的方式 二. 磁盘文件系统 2.1 磁盘的分区管理方法 2.2 文件名和inode的关系 三. 结合文件系统对文件创建和删除的相关问题的理解 3.1 文件创建时操作系统进行的工…

如何给合宙ESP32-C3刷写arduino固件,arduinoIDE的配置,测试代码

视频教程 https://github.com/Yu-1120/ESP32-C3 资料下载地址 合宙ESP32-C3刷写arduino固件 然后点击安装就可以了 arduino-IDE的配置 我用的版本&#xff1a;2.1.1&#xff08;版本不对也多大没关系&#xff09; 下载安装 选择 ESP32C3 Dev Module 安装环境 配置环境&am…

二十六、传输层协议(下)

一、滑动窗口 刚才我们讨论了确认应答策略&#xff0c;对每一个发送的数据段&#xff0c;都要给一个ACK确认应答. 收到ACK后再发送下一个数据段。这样做有一个比较大的缺点, 就是性能较差. 尤其是数据往返的时间较长的时候. 既然这样一发一收的方式性能较低, 那么我们一次发送…

snpEff注释结果解读

目录 1.帮助文档 1.1 常用参数 2. 命令的用法&#xff1a; 3. 结果文件解读 4. SNP下游的分析 利用snpEff软件对 snp.vcf &#xff08;利用gatk软件calling-snp&#xff09;进行注释&#xff0c;运行下述命令&#xff1a; ## 构建好物种的数据库 java -jar /opt/snpEff/s…

基于Spring Boot的扶贫助农商城系统设计与实现(Java+spring boot+MySQL+VUE)

获取源码或者论文请私信博主 演示视频&#xff1a; 基于Spring Boot的扶贫助农商城系统设计与实现&#xff08;Javaspring bootMySQLVUE&#xff09; 使用技术&#xff1a; 前端&#xff1a;html css javascript jQuery ajax thymeleaf 微信小程序 后端&#xff1a;Java spr…

QTranslator语言转换

//appname的格式 例如通常为&#xff08;QQ为应用的名称&#xff09; QQ_en.ts或QQ_zh_CN.ts QString qmName"zh_CN"; QTranslator trans ; QString qm QString(":/translatoin/qt/appname_%1.qm").arg(qmName); auto ret trans.load(qm); Q_UNUSED(ret)…

CSS高级特性

1.CSS复合选择器 CSS复合选择器&#xff1a;复合选择器是由两个或多个基础选择器通过不同的方式组合而成的 1.1 标签指定式选择器&#xff1a;又称交集选择器&#xff0c;由两个选择器构成&#xff0c;其中第一个选择器为标记选择器&#xff0c;第二个为class选择器或id选择器…

【Spring core学习一】简单认识Spring是什么?

目录 1、为什么要学习Spring&#xff1f; 2、Spring是什么&#xff1f; 1、IoC是什么&#xff1f; 2、进一步通过代码演示理解IoC 3、怎么理解容器&#xff1f; 4、知道DI与IoC的区别&#xff1f; 1、为什么要学习Spring&#xff1f; 我们常说的Spring 指的是 Spring Fra…

地平线旭日x3派40pin引脚控制,点亮小灯,控制舵机

地平线旭日x3派40pin引脚控制&#xff0c;点亮小灯&#xff0c;控制舵机 引脚对照表点亮RGB小灯安装旭日X3派WiringPi使用WiringPi点亮RGB小灯使用软件PWM功能 官方用户手册中只有python控制教程&#xff0c;没有c语言控制教程。且官方的教程中并没有软件pwm功能。本教程在开发…

Linux——动静态库的制作和使用(实操+代码+原理介绍)

动静态库的制作和使用 1️⃣.动静态库介绍&#x1f3c0;静态库⚽️动态库&#x1f3c8;区别&#x1f3d0;使用动态库的优点包括&#xff1a;&#x1f3c9; 使用静态库的优点包括&#xff1a; 2️⃣静态库的制作&#x1f34a;Q:库文件能不能有main()函数&#xff1f;&#x1f34…

imazing是什么软件?2023年imazing官网中文版下载

最近很小伙们&#xff0c;咨询兔八哥&#xff0c;imazing是什么软件&#xff1f;&#xff0c;今天兔八哥爱分享整理一下imazing到底是什么软件&#xff1f;好用吗&#xff1f; imazing是一款iOS设备管理软件,借助 iMazing 的独有 iOS 备份技术&#xff08;无线、隐私和自动&am…