搜索算法合集 - By DijkstraPhoenix

搜索算法合集

By DijkstraPhoenix

深度优先搜索 (DFS)

引入

如果现在有一个迷宫,如何走路径最短?

方法

走迷宫最简单粗暴的方法式什么呢?当然是把所有路都走一遍啦!

如果是手动计算的话,可能会把你手指累得抽筋,但电脑不会,电脑具有强大的算力,这种暴力的事情当然是交给电脑做啦。

深搜的本质:一条路走到底,走到死胡同再往回走,回到上一个岔口继续走,直到找到正确的路

实际上,任何一条路都可以看做是一个只有一个岔口的分岔路,所以不需要把路和岔口分开计算。

那么刚才的例子应该是这么走(数字代表第几次尝试)实际上岔口走的顺序是任意的,方法不唯一。

概念:从死胡同返回的步骤叫做回溯

由于深搜不能保证第一次找到的路径为最短路径,所以需要统计所有路线

深搜一般使用递归实现,走过的每个位置都要打上标记,同一条路不能再走一遍

主算法代码:

int maze[MAXN][MAXN];//存储迷宫 0表示当前节点可以走,1表示不能走
bool vis[MAXN][MAXN];//打标记
const int dx[]={1,0,-1,0};
const int dy[]={0,1,0,-1};//位移数组,分别对应 上右下左(如果是八向移动的话要改成对应的)
int n,m,stx,sty,edx,edy;//地图长宽以及起点和终点的坐标
int ans=0x7f7f7f7f;//最短距离,要初始化为极大值void dfs(int x,int y,int z)//x和y是当前位置的坐标,z是走过的步数
{if(x==edx&&y==edy)//到了终点{ans=min(ans,z);//更新答案(如果答案还是极大值,说明无法到达终点)return;}vis[x][y]=true;//打标记for(int i=0;i<4;i++)//枚举四个方向{int nx=x+dx[i],ny=y+dy[i];//下一个应该走到的位置if(nx<1||nx>n||ny<1||ny>m)continue;//不能走出地图(这个要写在灵魂拷问的最前面,否则访问数组要越界)if(maze[nx][ny]==1)continue;//不能卡墙里if(vis[nx][ny])continue;//不能走你走过的路dfs(nx,ny,z+1);//走到下一个节点}vis[x][y]=false;//重点!回溯时要清除标记!
}

例题

迷宫

洛谷 P1605

解法见上文

#include<iostream>
#include<cstring>
using namespace std;
int num=0;
int n,m,t,edx,edy,stx,sty;
int maze[10][10];
int vis[10][10];
int dx[]={0,1,0,-1};
int dy[]={1,0,-1,0};
void dfs(int x,int y)
{vis[x][y]=1;if(x==edx&&y==edy){num++;vis[x][y]=0;return;}for(int i=0;i<4;i++){int nx=x+dx[i];int ny=y+dy[i];if(nx<1||nx>n||ny<1||ny>m)continue;if(maze[nx][ny]==1)continue;if(vis[nx][ny]==1)continue;dfs(nx,ny);}vis[x][y]=0;
}
int main(void)
{int xjx,xjy;memset(vis,0,sizeof(vis));memset(maze,0,sizeof(maze));cin>>n>>m>>t;cin>>stx>>sty>>edx>>edy;for(int i=1;i<=t;i++){cin>>xjx>>xjy;maze[xjx][xjy]=1;}vis[stx][sty]=1;dfs(stx,sty);cout<<num;return 0;
}

八皇后问题

洛谷 P1219

本题的每一步都决定一个皇后的位置,由输出格式就可以看出,我们可以按每一列的顺序计算。一个皇后会独占一行、一列、两斜线,因为是按列计算的,不需要给列打标记,则需要 3 个标记数组。

(其实可以看一下洛谷上的题解)

#include<bits/stdc++.h>
using namespace std;
bool vis[15],vis1[35],vis2[35];
int n;
int nod[15];
int sum=0;
void dfs(int k)
{if(k>n){sum++;if(sum<=3)//前3个要输出方案{for(int i=1;i<=n;i++)cout<<nod[i]<<" ";cout<<endl;}return;}for(int i=1;i<=n;i++){if(vis[i])continue;if(vis1[i+k-1])continue;if(vis2[i-k+13])continue;vis[i]=true;vis1[i+k-1]=true;vis2[i-k+13]=true;//可以手动模拟一下行列坐标和斜坐标的关系,加13是防止计算出负数nod[k]=i;//保存方案dfs(k+1);vis[i]=false;vis1[i+k-1]=false;vis2[i-k+13]=false;}
}
int main(void)
{cin>>n;dfs(1);cout<<sum;return 0;
}

全排列问题

洛谷 P1706

按照题意模拟搜索即可

#include<iostream>
#include<cstdio>
using namespace std;
int n,a[1000],vis[1000];
void dfs(int step)
{if(step==n+1){for(int i=1;i<=n;i++){printf("%5d",a[i]);//题目要求格式化输出}cout<<endl;}for(int i=1;i<=n;i++){if(vis[i]==1)continue;a[step]=i;vis[i]=1;dfs(step+1);vis[i]=0;}
}
int main(void)
{cin>>n;dfs(1);return 0;
}

一些建议练习的题

求细胞数量
提示:联通块问题,不要清除标记,从每个未标记且是细胞的块出发,将整个块打上标记

小猫爬山

选数

单词接龙

--没写完呢--

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

个人知识面/技能池

虽然走得慢,但是一直在前行 知识面/技能池 电路 模拟电路微弱信号处理信号链设计1Msps采样电路设计滤波器设计无源滤波器有源滤波器光电探测电路设计电力电子逆变电路设计磁耦合谐振式无线电能传输开关电源LED恒流驱动AC/DC 设计基本电路知识电路基础知识复习跟习题册联系现代…

【CodeForces训练记录】Codeforces Round 977 (Div. 2, based on COMPFEST 16 - Final Round)

赛后反思 做红温了,太菜了,每题都需要WA几次才能过,B题看到 MEX 选择性害怕,时间复杂度又算错了 A题 每次选择一对 \(a_i,a_j\) 把均值插入数组最后面,要想结果最大,对于两个数求均值,最后的结果一定是小于等于其中的较大值,我们可以考虑如何最大化最后一次操作,想到将…

傻逼模拟赛搬的时候能不能看看题面改之后还是不是让人能看懂还有不发 checker 是有什么心事吗

如题。 傻逼模拟赛搬的时候能不能看看题面改之后还是不是让人能看懂还有不发 checker 是有什么心事吗还在最后一道题放集训队互测什么意思 什么叫有 \(b_{k}\) 种 \(k\) 类型的货币,同一种流通的货币不会超过二十种 什么叫接下来 \(n\) 个数表示 \(a_{1} \sim a_{n-1}\)upd:

Java - 10 二维数据

Java - 10 二维数据 一维数组的每个元素又是一个一维数组 静态初始化 int[][] arr = {{0,0,0,0},{1,1,1,1},{2,2,2,2},{3,3,3,3}};public class TwoDimensionArray {public static void main(String[] args) {int[][] arr = {{0,0,0,0},{1,1,1,1},{2,2,2,2},{3,3,3,3}};// 遍历…

Java - 11 类与对象

Java - 11 类与对象 类 类[属性, 行为] ->对象[属性, 行为] public class Test{public static void main(String[] args){Cat cat1 = new Cat(); // 创建对象cat1.name = "大宝";cat1.age = "3";cat1.color = "orange";System.out.println(ca…

20222413 2024-2025-1 《网络与系统攻防技术》实验一实验报告

1.实验内容 在本周的学习过程中,我了解到了许多缓冲区溢出攻击的实际案例、缓冲区溢出攻击的原理和相关基础知识,包括GDB调试器的使用方法、反汇编、基础的汇编语言与指令等,重新温习了函数调用过程和进程管理方面的知识内容。并且通过实验一,我能够了解并熟练完成Linux系统…

函数的上下文

函数的上下文 概述 在函数体的语句中,会出现this这个词,this就是函数的上下文 函数中this是谁,就说明函数的上下文是谁 函数中的this是谁,要看是如何调用的,因为this不是一成不变的 比如我们看下面的例子 var obj = {a: 100,fun: function() {console.log(this.a);} };我们…

拥挤聚集智能监测系统

拥挤聚集智能监测系统可以通过对人员数量、密度等进行实时监测,拥挤聚集智能监测系统识别出拥挤聚集的情况,并及时发出预警。拥挤聚集智能监测系统可以通过对人员进车间的人数等进行监测,识别出是否存在人员拥堵、挤压等安全隐患,及时发出警报,提醒工作人员采取措施疏散人…

睡岗识别 AI助力企业安全管控

睡岗识别可以通过AI视频智能分析技术,睡岗识别识别出操作人员是否存在睡岗情况。例如,在变电站等场景中,睡岗识别技术可以通过对识别出操作人员是否存在睡岗情况,及时发出预警,避免因操作人员的疏忽而导致的安全事故。在工厂车间中,睡岗识别技术可以通过对工人的行为进行…

加油站安全风险监测预警系统

加油站安全风险监测预警系统可以通过对加油站设备、环境、人员等方面进行监测,加油站安全风险监测预警系统实现对加油站的全面监管。例如,在加油站油罐区中,加油站安全风险监测预警系统可以对加油站人员抽烟打电话、明火烟雾等环境安全隐患进行自动识别,及时发出预警,避免…

山西煤矿电子封条

山西煤矿电子封条通过AI视觉分析技术,山西煤矿电子封条实现对各矿区(煤矿和非煤矿区)每日矿井出入井人监察控制、调度室空岗识别、煤矿生产作业状态、摄像头遮挡、挪动角度识别、货运车辆出矿识别等。山西煤矿电子封条实现当前待办事项的推送,以及对各矿区用户区域内的报警…

离岗识别 AI助力企业安全管控

离岗识别通过yolov5网络模型技术,离岗识别可以自动识别办公室、工厂、监控室监控画面中人员离岗脱岗睡岗等行为,发现违规行为立即抓拍告警并同步睡岗离岗等违规数据到后台提醒值班人员及时处理。离岗识别采用人工智能算法识别技术对各主控室、办公室、工厂、煤矿监控室等人员…