C++ 双向广度搜索,嚯嚯!不就是双指针理念吗

1. 前言

在线性数据结构中搜索时,常使用线性搜索算法,但其性能偏低下,其性能改善方案常有二分搜索和双指针或多指针搜索算法。在复杂的数据结构如树和图中,常规搜索算法是深度和广度搜索。在深度搜索算法过程中常借助剪枝或记忆化方案提升搜索性能。广度搜索算法过程中常见的性能优化方案为双向广度搜索和启发式搜索。双向广度搜索可以认为是图论中的双指针搜索方案,本文将和大家深入探讨其算法细节。

图中常见的操作为最短路径查找。如在下面的无向无权重图中查找节点1到节点6之间的最短路径,可以直接使用广度搜索算法找到。

1.png

无向无权重图中,直接使用广度搜索算法查找节点之间的最短路径的基本模板代码:

#include <iostream>
#include <queue>
using namespace std;
//邻接矩阵
int graph[100][100];
//记录是否被访问过
int vis[100];
//边数与顶点数
int n ,m;
//节点距离起始点的最短路径
int dis[100];
void init() {
	for(int i=1; i<=n; i++) {
		for(int j=1; j<=n; j++) {
			graph[i][j]=0;
		}
		dis[i]=0;
		vis[i]=0;
	}
}
void addEdge() {
	int f,t;
	for(int i=1; i<=m; i++) {
		cin>>f>>t;
		graph[f][t]=1;
		graph[t][f]=1;
	}
}

//广度搜索
void bfs(int start,int end) {
	//队列
	queue<int> myq;
	//初始化队列
	myq.push(start);
	vis[start]=1;
	dis[start]=0;
	while( !myq.empty() ) {
		int size=myq.size();
		//找到队列中的所有节点
		for(int i=0; i<size; i++) {
			int t= myq.front();
			myq.pop();
			//搜索到终点 
			if(t==end)return;
			//扩展所有子节点入队列
			for(int j=1; j<=n; j++) {
				if(graph[t][j]==1 &&  vis[j]==0) {
					myq.push(j);
					vis[j]=1;
					dis[j]=dis[t]+1;
				}
			}
		}
	}
}
void show() {
	for(int i=1; i<=n; i++)
		cout<<i<<"-"<<dis[i]<<"\t";
}
int main(int argc, char** argv) {
	cin>>n>>m;
	init();
	addEdge();
	bfs(1,n);
	show();
	return 0;
}

测试结果:

2.png

当图中的节点很多,关系较复杂时,直接使用广度搜索其时间复杂度非常大。对于上述问题,既然已经知道了起点和终点,可以使用类似于双指针的方案,让搜索分别从起点和终点开始,从两端相向进行。这样可以减少一半的搜索量,此种搜索方案称为双向广度搜索。

2. 初识双向广度搜索

不是任何时候都可以使用双向广度搜索,只有当起点和终点已知情况方可使用。如下图所示,从起点向终点方向的搜索称为正向搜索,从终点向起点方向的搜索称为逆向搜索。

3.png

下面演示双向搜索的过程。

  • 双向广度搜索实现过程中,可以使用2个队列,也可以仅使用1个队列。这里先使用 2个队列的方案。正向搜索方向的队列命名为q1,逆向搜索方向的队列命名为q2

4.png

  • 初始化2个队列。q1中压入起点,q2中压入终点。

5.png

  • 扩展方案。如果q1的尺寸小于q2,则扩展q1队列,否则,扩展q2队列。初始q1q2两个队列的尺寸相同,此时可以选择扩展q1队列。扩展后的结果如下图所示。

6.png

  • 下面扩展q2 队列。

7.png

  • 至此,继续扩展q1队列。发现节点3和节点2的子节点4、5已经被访问过,且存放在q2中,可以此判断双向搜索相遇,可以认定双向搜索结束。
    15.png

对于相遇条件总结一下。

当对一个队列中的节点进行扩展时,发现此节点的子节点已经被另一个搜索队列扩展,可以认定两个搜索过程相遇。类似于两个施工队相向方向挖一条壕沟时,两者一定是相遇到对方挖通的位置,也就是说当一方挖到了对方正在挖的位置。

也可以在搜索结束后,查看那一个队列不为空,不空的队列中的节点即为相遇的节点。

如面使用代码描述上述的整个流程。

#include <iostream>
#include <queue>
using namespace std;
//邻接矩阵
int graph[100][100];
//边数与顶点数
int n ,m;
//正向搜索时,节点距离起始点的最短路径,也可以记录节点是否被访问过
int dis[100];
//逆向搜索时,节点距离终点的最短路径
int dis_[100];
//初如化
void init() {
	for(int i=1; i<=n; i++) {
		for(int j=1; j<=n; j++) {
			graph[i][j]=0;
		}
        //因为节点和自己的距离为 0,用-1 表示没有被访问,
		dis[i]=-1;
		dis_[i]=-1;
	}
}
//构建图
void addEdge() {
	int f,t;
	for(int i=1; i<=m; i++) {
		cin>>f>>t;
		graph[f][t]=1;
		graph[t][f]=1;
	}
}

//广度搜索
int bfs(int start,int end) {
	//正向队列
	queue<int> q1;
	//逆向队列
	queue<int> q2;
	//初始化队列
	q1.push(start);
	dis[start]=0;
	q2.push(end);
	dis_[end]=0;
     //任意队列为空时结束
	while( !q1.empty() && !q2.empty() ) {
         //计算队列的尺寸
		int s1=q1.size();
		int s2=q2.size();
		if(s1<=s2) {
			//扩展 q1
			for(int i=0; i<s1; i++) {
				int t= q1.front();
				q1.pop();
				//扩展所有子节点入队列
				for(int j=1; j<=n; j++) {
                      //不是子节点或者被访问过都跳过
					if(graph[t][j]==0 ||  dis[j]!=-1)continue;
                      //入队
					q1.push(j);
                      //计算距离
					dis[j]=dis[t]+1;
                      //如果出现在对方队列中,搜索结束
					if( dis_[j]!=-1)return dis[t]+dis_[j]+1;
				}
			}
		} else {
			//扩展 q2
			for(int i=0; i<s2; i++) {
				int t= q2.front();
				q2.pop();
				//扩展所有子节点入队列
				for(int j=1; j<=n; j++) {
					if(graph[t][j]==0 ||  dis_[j]!=-1)continue;
					q2.push(j);
					dis_[j]=dis_[t]+1;
					if(  dis[j]!=-1)return dis_[t]+dis[j]+1;
				}
			}
		}
	}
}
int main(int argc, char** argv) {
	cin>>n>>m;
	init();
	addEdge();
	int d= bfs(1,n);
	cout<<d;
	return 0;
}

也可以使用一个队列实现双向搜索算法。下面演示使用一个队列实现双向搜索流程。为了区分节点是属于正向还逆向搜索到的节点,用两种颜色分别表示,红色表示正向搜索到的节点,绿色表示逆向搜索到的节点。

8.png

  • 初始化队列。把起点和终点分别压入队列中。

9.png

  • 按正常流程对队列中的节点进行扩展。如下,扩展节点6的子节点4、5入队列。

10.png

  • 继续扩展节点1的子节点2、3入队列。

11.png

  • 继续扩展时,发现需要扩展的子节点已经存在于队列中,说明,已经相遇了。

12.png

3. 深度理解

下面通过几个案例让大家更深入的理解双向广度搜索。

3.1 字串变换,

题目来自于https://www.luogu.com.cn/problem/P1032

题目描述

已知有两个字串 A、B 及一组字串变换的规则(至多 6 个规则),形如:

  • A1->B1
  • A2->B2

规则的含义为:在 A 中的子串 A1 可以变换为 B1A2 可以变换为 B2……。

例如:A=abcdB=xyz

变换规则为:

  • abc->xu,ud->y,y->yz

则此时,A 可以经过一系列的变换变为 B,其变换的过程为:

  • abcd->xud->xy->xyz

共进行了 3 次变换,使得 A 变换为 B

输入格式

第一行有两个字符串 A,B

接下来若干行,每行有两个字符串 Ai,Bi,表示一条变换规则。

输出格式

若在 10 步(包含 10 步)以内能将 A 变换为 B,则输出最少的变换步数;否则输出 NO ANSWER!

样例 #1

样例输入 #1

abcd xyz
abc xu
ud y
y yz

样例输出 #1

3

算法分析:

根据样例绘制转换流程图。

13.png

虽然根据样例绘制出来是线性数据结构,但因规则可以很多,如果再添加如下几条新规则,则转换流程图就可能是图结构。

  • cd->z
  • ab->xy

14.png

以最大可能性考虑此题,其转换过程就是一个无向无权重图结构,且本质就是在图中查找起点到终点的最短路径。可以直接使用BFS算法,当数据量较大时,可以使用双向BFS搜索算法。下面代码使用双向广度搜索方案。

#include <iostream>
#include <queue>
#include <map>
using namespace std;
//开始和结束字符串
string s,e;
//存储转换规则
map<string,string> rule;
//存储节点至起点的距离
map<string,int> sdis;
//存储节点至能终点的距离
map<string,int>  edis;
//记录规则数量
int size=0;

int extend(queue<string> &q,map<string,int> &dis,map<string,int> &dis_,int type) {
	//队列的大小
	int size=q.size();
	//遍历队列
	for(int i=0; i<size; i++) {
		string t=q.front();
		string t1=t;
		q.pop();
		//查找可以转换的状态
		map<string,string>::iterator begin=rule.begin();
		map<string,string>::iterator end=rule.end();
		while(begin!=end) {
			string first= begin->first,second=begin->second;
			//如果是逆向搜索,规则也要改成逆向 
			if(type)first=begin->second,second=begin->first;
			//查找是否有可以转换的子串
			int pos= t1.find(  first );
			if(pos!=-1) {
				//得到可转换字符串
				t1.replace(pos,first.size(),second);
				//已经访问过
				if( dis[t1] ) continue;
				//没有访问过
				q.push(t1);
				dis[t1]=dis[t]+1;
				if(dis_[t1] )
					return dis_[t1]+dis[t]-1;
				t1=t;
			}
			begin++;
		}
	}
	return 0;
}

int bfs() {
	//正向队列
	queue<string> zxq;
	//反向队列
	queue<string> fxq;
	//初始化队列
	zxq.push(s);
	fxq.push(e);
	//即表示距离也表示访问过 
	sdis[s]=1; 
	edis[e]=1;
	int res=0;

	while( !zxq.empty() &&  !fxq.empty() ) {
		int size1=zxq.size();
		int size2=fxq.size();
		if( size1<=size2 ) res= extend(zxq,sdis,edis,0); //扩展正向队列
		else res=  extend(fxq,edis,sdis,1); //扩展反向队列
		if(res!=0)break;
	}

	return res;
}


int main() {
	cin>>s>>e;
	string f,t;
	while(cin>>f>>t) {
		rule[f]=t;
		size++;
	}
	int res=bfs();
	cout<<res;
	return 0;
}

3.2 八数码问题

题目描述:

八数码问题是典型的状态图搜索问题。在一个3×3的棋盘上放置编号为1~88个方块,每个占一格,另外还有一个空格。与空格相邻的数字方块可以移动到空格里。

任务1:指定初始棋局和目标棋局,计算出最少的移动步数;

任务2:输出数码的移动序列。

此题可以使用双向广度搜索算法查找到结果。因为正向和逆向搜索的扩展数量是相同的,可以使用一个队列实现,且正向搜索过的节点状态用1表示,逆向搜索过的节点状态用2表示。当节点和子节点的状态值之和为 3的时表示当正向和逆向搜索相遇。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;

int ed=123804765,st,f[10][5]={{0,1},{1,0},{-1,0},{0,-1}},s[5][5];
map<int,int> vis,d;
int bfs(int a,int b)
{
    queue<int> q;
    //正向搜索状态设置为 1
    vis[a]=1,d[a]=0;
    //逆向搜索状态设置为 2
    vis[b]=2,d[b]=0;
    //入队列
    q.push(a);
    q.push(b);
    while(!q.empty())
    {
        int u=q.front();
        q.pop();
        int v=u,x,y;
        //将数放入二维数组
        for(int i=3;i>=1;i--)
            for(int j=3;j>=1;j--)
            {
                s[i][j]=v%10;//分解数字
                v/=10;
                if(!s[i][j]) x=i,y=j;//找出0的位置
            }
        //将0进行移动
        for(int i=0;i<4;i++)
        {
            //向四周搜索
            int sx=x+f[i][0],sy=y+f[i][1];
            //越界检查
            if(sx<1||sx>3||sy<1||sy>3) continue;
            //得到新状态
            swap(s[x][y],s[sx][sy]);
            v=0;//还原成数字状态
            for(int i=1;i<=3;i++)
                for(int j=1;j<=3;j++)
                    v=v*10+s[i][j];
            if(vis[v]==vis[u])
            {
                //如果已经访问过
                swap(s[x][y],s[sx][sy]);
                continue;
            } 
            //如果相遇
            if(vis[v]+vis[u]==3) return d[u]+1+d[v];//起点为1,终点为2,相加为3
            d[v]=d[u]+1;
            vis[v]=vis[u];
            q.push(v);
            swap(s[x][y],s[sx][sy]);
        }
    }
    return -1;
}
int main ()
{
    cin>>st;
    if(st==ed) cout<<0<<"\n";
    else cout<<bfs(st,ed)<<"\n";
    return 0;
}

4. 总结

本文讲解了双向广度搜索算法,和双指针算法一样,让搜索双向同时进行,可以减沙近一半的搜索范围,提升搜索性能。记住,双向搜索算法要求在已知起点和终点的条件方可使用。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

分布式文件系统 SpringBoot+FastDFS+Vue.js【二】

分布式文件系统 SpringBootFastDFSVue.js【二】 六、实现上传功能并展示数据6.1.创建数据库6.2.创建spring boot项目fastDFS-java6.3.引入依赖6.3.fastdfs-client配置文件6.4.跨域配置GlobalCrosConfig.java6.5.创建模型--实体类6.5.1.FastDfsFile.java6.5.2.FastDfsFileType.j…

Mac M2芯片配置PHP环境

Mac M2芯片配置PHP环境 1. XAMPP 1. XAMPP 官网地址 https://www.apachefriends.org/ 安装 安装完成 web server打开后&#xff0c;在打开localhost 成功&#xff01;

(三十九)大数据实战——Prometheus监控平台的部署搭建

前言 Prometheus监控&#xff08;Prometheus Monitoring&#xff09;是一种开源的系统监控和警报工具。它最初由SoundCloud开发并于2012年发布&#xff0c;并在2016年加入了云原生计算基金会&#xff08;CNCF&#xff09;。Prometheus监控旨在收集、存储和查询各种指标数据&am…

Android---DslTabLayout实现底部导航栏

1. 在 Android 项目中引用 JitPack 库 AGP 8. 根目录的 settings.gradle dependencyResolutionManagement {...repositories {...maven { url https://jitpack.io }} } AGP 8. 根目录如果是 settings.gradle.kts 文件 dependencyResolutionManagement {...repositories {...…

A. Desorting

链接 : Problem - A - Codeforces 题意 : 思路 : 先判断序列是否排好序 &#xff0c; 不是排好序的&#xff0c;直接输出0即可&#xff0c;排好序的 : 先求出相邻元素之间的最小间隔&#xff0c;因为&#xff0c;要使有序非递减序列&#xff0c;变得不排序&#xff0c;…

OLMo 以促进语言模型科学之名 —— OLMo Accelerating the Science of Language Models —— 全文翻译

OLMo: Accelerating the Science of Language Models OLMo 以促进语言模型科学之名 摘要 语言模型在自然语言处理的研究中和商业产品中已经变得无所不在。因为其商业上的重要性激增&#xff0c;所以&#xff0c;其中最强大的模型已经闭源&#xff0c;控制在专有接口之中&#…

Leetcode-1572. 矩阵对角线元素的和

题目&#xff1a; 给你一个正方形矩阵 mat&#xff0c;请你返回矩阵对角线元素的和。 请你返回在矩阵主对角线上的元素和副对角线上且不在主对角线上元素的和。 示例 1&#xff1a; 输入&#xff1a;mat [[1,2,3],[4,5,6],[7,8,9]] 输出&#xff1a;25 解释&#xff1a;对角线…

Apache httpd 换行解析漏洞复现(CVE-2017-15715)

Web页面&#xff1a; 新建一个一句话木马&#xff1a; 0.php <?php system($_GET[0]); ?> 上传木马&#xff0c; burpsuite 抓包。 直接上传是回显 bad file。 我们查看数据包的二进制内容&#xff08;hex&#xff09;&#xff0c;内容是以16进制显示的&#xff0c;…

挑战杯 wifi指纹室内定位系统

简介 今天来介绍一下室内定位相关的原理以及实现方法; WIFI全称WirelessFidelity&#xff0c;在中文里又称作“行动热点”&#xff0c;是Wi-Fi联盟制造商的商标做为产品的品牌认证&#xff0c;是一个创建于IEEE 802.11标准的无线局域网技术。基于两套系统的密切相关&#xff…

html的列表标签

列表标签 列表在html里面经常会用到的&#xff0c;主要使用来布局的&#xff0c;使其整齐好看. 无序列表 无序列表[重要]&#xff1a; ul &#xff0c;li 示例代码1&#xff1a; 对应的效果&#xff1a; 无序列表的属性 属性值描述typedisc&#xff0c;square&#xff0c;…

U盘重装系统

因为系统管理员密码忘记&#xff0c;登录不了window系统&#xff0c;使用老毛桃制作U盘启动盘 1、下载老毛桃 下载地址为http://lmt.psydrj.com/index.html 安装后&#xff0c;桌面上显示为 2、制作U盘启动盘 启动老毛桃U盘启动装机工具&#xff0c;插入U盘&#xff0c;点击一…

[Java][算法 滑动窗口]Day 03---LeetCode 热题 100---08~09

第一题 无重复字符串的最长子串 思路 其实就是在字符串S中 找到没有重复的最长子串的长度 这道题的难点就是在于如何判断最长并且无重复 首先 最长长度 可以使用变量max记录保存 再者 判断有无重复 最简单的方法就是 暴力遍历法 即对于每次找的子串都再次寻找遍历…

【十九】【C++】 priority_queue简单使用和仿函数

priority_queue文档介绍翻译 优先队列是一种容器适配器&#xff0c;专门设计成其中的第一个元素始终是根据某种严格的弱排序准则最大的元素。 这种上下文类似于堆&#xff0c;其中元素可以在任何时刻插入&#xff0c;而只能检索最大堆元素&#xff08;在优先队列中顶部的元素&a…

为自监督学习重构去噪扩散模型

在这项研究中&#xff0c;作者检验了最初用于图像生成的去噪扩散模型&#xff08;DDM&#xff09;的表示学习能力。其理念是解构DDM&#xff0c;逐渐将其转化为经典的去噪自动编码器&#xff08;DAE&#xff09;。这一解构过程让大家能够探索现代DDM的各个组成部分如何影响自监…

【Docker】Docker Container操作案例 | 综合实战

文章目录 Docker Container操作案例容器的基本操作容器状态迁移容器批量处理技巧容器交互模式attached模式detached模式interactive模式 容器与宿主机内容复制容器自动删除容器自动重启容器环境变量设置容器详情查看容器执行单行命令容器镜像导入导出容器日志查看容器资源查看 …

C++:多态

C&#xff1a;多态 虚函数虚函数语法虚函数重写协变接口继承 多态构成成员函数状态对比抽象类多态原理多继承与多态虚继承与多态 先看到多态的定义&#xff1a; C的多态是指在面向对象程序设计中&#xff0c;允许使用基类的指针或引用来调用派生类的虚函数的特性。这样的调用将…

数据结构-并查集

并查集原理 在一些应用问题中&#xff0c;需要将n个不同的元素划分成一些不相交的集合。开始时&#xff0c;每个元素自成一个 单元素集合&#xff0c;然后按一定的规律将归于同一组元素的集合合并。在此过程中要反复用到查询一 个元素归属于那个集合的运算。适合于描述这类…

阿里云幻兽帕鲁服务器配置4核16G10M带宽够8个人玩吗?玩起来流畅度怎么样?

阿里云幻兽帕鲁服务器配置4核16G10M带宽这个&#xff0c;个人实测下来&#xff0c;五六个人玩是比较流畅的&#xff0c;不过8个人的话&#xff0c;估计会有点卡。如果是8个人的话&#xff0c;我建议选择8核32G那个配置&#xff0c;更加适合一些。 阿里云一键部署幻兽帕鲁详细教…

【lesson57】信号量和生产者消费者模型(环形队列版)

文章目录 信号量概念信号量接口初始化销毁等待发布 基于环形队列的生产者消费者模型编码Common.hLockGuard.hppTask.hppsem.hppRingQueue.hppConProd.cc 信号量概念 POSIX信号量和SystemV信号量作用相同&#xff0c;都是用于同步操作&#xff0c;达到无冲突的访问共享资源目的…

漫漫数学之旅022

文章目录 经典格言数学习题古今评注名人小传- 刘易斯卡罗尔 经典格言 艾丽斯笑着说&#xff1a;“去尝试也毫无用处&#xff0c;一个人无法相信不可能的事情。”——刘易斯卡罗尔&#xff08;Lewis Carroll&#xff09;《艾丽斯梦游仙境&#xff08;Alice in Wonderland&#…