Java数据结构《队列和邻接矩阵实现图的广度优先搜索》题目

一、前言:

  这是怀化学院的:Java数据结构中的一道难度偏难的一道编程题(此方法为博主自己研究,问题基本解决,若有bug欢迎下方评论提出意见,我会第一时间改进代码,谢谢!) 后面其他编程题只要我写完,并成功实现,会陆续更新,记得三连哈哈! 所有答案供参考,不是标准答案,是博主自己研究的写法。(这一个题书上也有现成的代码,重要的是理解它的算法原理!)

二、题目如下:(注意:题目有一个要求(难点之一):若一个顶点有两个邻接顶点,那么下一个访问顶点的顺序要按照:A -> Z 的字典顺序访问!)

(第 2 题) 图的广度优先搜索(难度系数100)

图的广度优先搜索
描述:
图的广度优先搜索类似于树的按层次遍历,即从某个结点开始,先访问该结点,然后访问该结点的所有邻接点,再依次访问各邻接点的邻接点。如此进行下去,直到所有的结点都访问为止。在该题中,假定所有的结点以“A”--“Z”中的若干字符表示,
且要求结点的访问顺序要求根据由“A”至“Z”的字典顺序进行访问。例如有如下图:

如果要求从H开始进行广度优先搜索,则搜索结果为:H->A->E->K->U.
输入:
输入只包含一个测试用例,第一行为一个自然数n,表示顶点的个数,第二行为n个大写字母构成的字符串,表示顶点,接下来是为一个n*n大小的矩阵,表示图的邻接关系。数字为0表示不邻接,否则为相应的边的长度。
最后一行为一个字符,表示要求进行广度优先搜索的起始顶点。
输出:
用一行输出广度优先搜索结果,起始点为给定的顶点,各顶点之间用一个空格隔开。要求同一顶点的邻接点的访问顺序按“A”---“Z”的字典顺序。
样例输入:
5
HUEAK
0 0 2 3 0
0 0 0 7 4
2 0 0 0 0
3 7 0 0 1
0 4 0 1 0
H
样例输出:
H A E K U

三、代码实现:(代码的做题原理全部在代码注释中,若还有疑问也可以翻书) 

补充:这里我用到的就是队列去实现广度优先算法,因为队列先进先出,先进队列的先访问,然后再相邻的顶点后进去后访问,依次下来就实现成功!

(1)我创建了最基本的图类,里面有它的很多成员变量与成员的操作方法。(所以代码会很长,但是解释了基本原理。实际上的图的广度优先遍历算法就只有一个方法在里面)

package com.fs.graph;import java.util.ArrayList;
import java.util.Collections;   //用到工具类Collections
import java.util.LinkedList;   //要用到LinkedListed是实现类
import java.util.Queue;   //要用到队列接口,导包public class Graph_01 {//为了安全性,全部用private修饰封装,要使用或者赋值只能使用构造方法或者getter、setter方法private ArrayList<String> arrayList;  //这里我用这个ArrayList集合来代替数组去存储输入的每个顶点,并设置集合里面存储的元素是String类型的private int[][] matrix;  //这个二维数组:用来存储输入的矩阵,表示各个顶点相邻边的关系private int vertex;  //用来表示顶点的实际个数private int maxVertex;  //用来表示顶点的最大数量private boolean[] visit;  //用来判断每个顶点是否被访问过public Graph_01(int vertex){this.maxVertex=10;  //默认设置最大顶点个数为10this.arrayList = new ArrayList<>();this.vertex=vertex;}public ArrayList<String> getArrayList() {return arrayList;}//定位顶点的位置public int indexOfVex(String v){for(int i=0;i<vertex;i++){if(arrayList.get(i).equals(v)){return i;  //找到该顶点在集合中的位置}}return -1;  //否则返回-1,表示没找到该顶点}//传入包含所有顶点的字符串public void insertVer(String vertex01) throws VertexException{if(vertex01.length()>maxVertex){throw new VertexException ("超出最大顶点数!");}//存入顶点for(int i=0;i<vertex;i++){String str = vertex01.charAt(i)+"";arrayList.add(str);}}//存入顶点间边的邻接关系矩阵public void addEdge(int [][] matrix01){this.matrix=matrix01;}//实现图的广度优先搜索算法public String broadFirstSearch(int v) throws VertexException{if(v<0||v>maxVertex){//如果这个顶点不在集合中,抛出异常throw new VertexException("该顶点不存在!");}visit = new boolean[vertex];  //初始化:判断顶点是否被访问数组StringBuilder sb = new StringBuilder();  //用这个方便字符串的添加和删除,因为不会产生多余的对象,造成内存浪费,还有这个没有实现线程安全功能,效率更高//队列接口是继承Collection接口,它的实现类就有:LinkedList类Queue<Integer> queue = new LinkedList<>();  //说明里面操作时的元素类型是整型的queue.offer(v);  //先把第一个访问的顶点进入队列中visit[v]=true;  //并标记该顶点的访问状态为:true,表示访问过//只要队列没空就是所有的顶点还未访问完while(!queue.isEmpty()){  //判断队列是否为空v=queue.poll();  //进队列后登记完访问状态然后出队列sb.append(arrayList.get(v)+" ");  //将遍历完的顶点加入字符串缓冲区ArrayList<String> arrayList01= new ArrayList<>();for(int i=vertex-1;i>=0;i--) {//邻接关系矩阵某一行中只要有不为0的值,那就是属于v顶点的这一行,还有另外有顶点与它有关系//而且要没被访问过,或者不是大于整型的最大值(表示无穷:无边的关系)if ((matrix[v][i] != 0) && (!visit[i]) && (matrix[v][i] != Integer.MAX_VALUE)) {arrayList01.add(arrayList.get(i));  //将所有满足有关系的顶点全部放在一个集合里.为了满足题目要按照字典顺序访问才这么写的}}//先把某个顶点所有的邻接顶点按照字典顺序排好序Collections.sort(arrayList01);//为了满足条件:就需要字典顺序在前的顶点先入队列,先出队先被访问。而字典顺序在后的顶点后入队列,后出队列后被访问(这样就实现了题目要求)for(int j=0;j<arrayList01.size();j++){//先从字典排序在前的顶点开始操作String str=arrayList01.get(j);  //获取该顶点int index=indexOfVex(str);  //再找到该顶点在之前顶点集合的位置queue.offer(index);  //最后把该顶点的入队visit[indexOfVex(str)]=true;   //并设置已经访问过}}//用问号表达式,如果满足前面的就输出问号后面的,否则就是输出nullreturn sb.length()>0?sb.substring(0,sb.length()):null;  //用这个substring()方法,可以将StingBuilder里的字符串缓冲区里的字符串转换成String类型的}
}
//自定义异常类(考试可以不写,没做要求)
class VertexException extends Exception{public VertexException(){super();}public VertexException(String message){super(message);}
}

(2)根据题目要求输入创建了测试类:

package com.fs.graph;import java.util.Scanner;public class Test_Graph_01 {public static void main(String[] args) {Scanner sc =new Scanner(System.in);int n= sc.nextInt();  //代表顶点个数String vertex = sc.next();  //代表的所有顶点的字符串Graph_01 graph = new Graph_01(n);try {graph.insertVer(vertex);  //输入由各个顶点构成的字符串}catch (VertexException e){e.printStackTrace();return;}int [][] matrix01 = new int[n][n];for(int i=0;i<n;i++){for(int j=0;j<n;j++){matrix01[i][j]=sc.nextInt();}}graph.addEdge(matrix01);  //传入邻接关系矩阵String v = sc.next();  //传入从某一个指定的顶点去遍历图try{String path = graph.broadFirstSearch(graph.indexOfVex(v));  //先传入起始遍历顶点,再定位其位置,最后调用广度优先算法System.out.print(path);  //输出最终遍历结果}catch (VertexException e){e.printStackTrace();}}
}

四、不同情况的代码测试运行结果:

<1>首先是题目中的测试输入样例:(最好手打输入测试,直接复制可能格式问题导致报错!)

 <2>自己测试的输入样例: (测试7个顶点!)

1.图展示:(手写比较随意,字有点丑哈哈!)

2.也就是测试输入样例为: 

(1.)从H顶点开始遍历图:(提示最好不要直接复制我的输入样例,最好用手跟着打。因为复制完格式有点问题,就会报错)

7
HAEBKUD
0 3 2 1 0 0 0 
3 0 0 0 1 7 0
2 0 0 0 0 0 0
1 0 0 0 0 0 1
0 1 0 0 0 4 0
0 7 0 0 4 0 0
0 0 0 1 0 0 0
H

代码运行后输出的遍历结果:

(2.)从D顶点开始遍历图:(提示最好不要直接复制我的输入样例,最好用手跟着打。因为复制完格式有点问题,就会报错) 

7
HAEBKUD
0 3 2 1 0 0 0 
3 0 0 0 1 7 0
2 0 0 0 0 0 0
1 0 0 0 0 0 1
0 1 0 0 0 4 0
0 7 0 0 4 0 0
0 0 0 1 0 0 0
D

代码运行后输出的遍历结果:

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

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

相关文章

redis相关题

1 什么是Redis Redis(Remote Dictionary Server) 是⼀个使⽤ C 语⾔编写的&#xff0c;开源的&#xff08;BSD许可&#xff09;⾼性能⾮关系型&#xff08;NoSQL&#xff09;的键值对数据库。Redis 可以存储键和五种不同类型的值之间的映射。键的类型只能为字符串&#xff0c;…

mysql相关查询语法(自用)

mysql 条件 if WHERE IF(条件, true执行条件, false执行条件 ) SELECT * FROM book WHERE IF(price > 120, btypeid10, btypeid11);case when SELECTpp_.id,pp_.project_name FROMpv_project pp_CASE 1WHEN trueTHEN1 1ELSEpp_.project_name bbbbb END 日期相关 …

C++作业3

设计一个Per类&#xff0c;类中包含私有成员:姓名、年龄、指针成员身高、体重&#xff0c;再设计一个Stu类&#xff0c;类中包含私有成员:成绩、Per类对象p1&#xff0c;设计这两个类的构造函数、析构函数和拷贝构造函数。 代码&#xff1a; #include <iostream>using n…

Flutter页面刷新失败?看看是不是这个原因

文章目录 问题描述解决办法在控件A中定义回调函数在页面中使用控件A 原因分析回顾问题原因分析 setState使用注意事项上下文正确性异步更新避免深层嵌套避免频繁调用避免在 build 方法中调用避免在 dispose 方法中调用 问题描述 我用flutter开发了一个页面&#xff0c;页面上有…

Java SpringBoot Controller常见写法

文章目录 环境Controller调用脚本运行结果总结 环境 系统: windows 11 工具: java, idea, git bash Controller 接口常见有以下几种方式 其中&#xff1a; Tobj 调用脚本 我的是windows 系统&#xff0c;使用 git bash 窗口运行, 用 cmd 或者 power shell 会有问题 curl …

8、Qt中定时器的使用

一、说明 在Qt中常使用如下两种定时器 1、使用QObiect类的定时器事件QTimerEvent 与定时器相关的函数有&#xff1a;startTimer()、timeEvent()、killTimer()&#xff1b;startTimer(int interval)函数开始一个定时器并返回定时器ID&#xff0c;如果不能开始一个定时器&…

手把手教你Spring Security Oauth2自定义授权模式

目录 前言1、自定义认证对象2、自定义TokenGranter3、自定义AuthenticationProvider4、配置自定义AuthenticationProvider、自定义TokenGranter5、配置客户端授权模式6、测试 前言 在Oauth2中&#xff0c;提供了几种基本的认证模式&#xff0c;有密码模式、客户端模式、授权码…

传统算法:使用 Pygame 实现选择排序

使用 Pygame 模块实现了选择排序的动画演示。首先,它生成一个包含随机整数的数组,并通过 Pygame 在屏幕上绘制这个数组的条形图。接着,通过选择排序算法对数组进行排序,动画效果可视化每一步的排序过程。在排序的过程中,程序找到未排序部分的最小元素,并将其与未排序部分…

数据结构 / 队列 / 循环队列

1. 定义 为充分利用向量空间&#xff0c;克服假溢出现象的方法是&#xff1a;将向量空间想象为一个首尾相接的圆环&#xff0c;并称这种向量为循环向量。存储在其中的队列称为循环队列&#xff08;Circular Queue&#xff09;。循环队列是把顺序队列首尾相连&#xff0c;把存储…

前端知识笔记(十二)—————前端面试容易问到的问题总结

1.$(document).ready()方法和window.onload有什么区别&#xff1f; 执行时间不同&#xff1a;window.onload必须等到页面内的所有元素&#xff08;&#xff09;加载完毕后才能执行。 $(document).ready()是页面DOM结构绘制完毕后就执行&#xff0c;不必等到加载完毕 执行次数…

字符集与编码规则

字符集 强调&#xff1a;UTF-8是编码规则&#xff0c;不是字符集 过程&#xff1a; 字符 --查表获得对应数字&#xff0c;--编码 解码---查表----获取字符 ASCII码 &#xff1a;一个字节 8bit GBK字符集&#xff08;windows系统默认使用的GBK,系统显示ANSI&#xff09; 存…

西南科技大学信号与系统A实验一(信号的产生与时域运算)

目录 一、实验目的 二、实验原理 三、实验内容 四、思考题 一、实验目的 1、 掌握用matlab软件产生基本信号的方法。 2、 应用matlab软件实现信号的加、减、乘、反褶、移位、尺度变换及卷积运算 二、实验原理 (一) 产生信号波形的方法 利用Matlab软件的信号处…

QLabel实现点击事件

主要是通过安装过滤器&#xff0c;实现点击事件。 事件过滤器的操作&#xff1a; 首先给需要添加事件过滤器的部件注册监听对象&#xff1b; 对象名->installEventFilter(true); 重写eventFilter(QObject *obj, QEvent *event)函数进行处理。 MainWindow::MainWindow(QW…

Java泛型核心知识总结

目录 Java 泛型核心知识总结泛型什么是泛型&#xff1f;有什么用&#xff1f;泛型有哪些限制&#xff1f;为什么&#xff1f;项目中哪里用到了泛型&#xff1f;什么是类型擦除&#xff1f;什么是桥方法&#xff1f; 通配符什么是通配符&#xff1f;有什么作用&#xff1f;通配符…

四、Lua循环

文章目录 一、while(循环条件)二、for&#xff08;一&#xff09;数值for&#xff08;二&#xff09;泛型for&#xff08;三&#xff09;repeat util 既然同为编程语言&#xff0c;那么控制逻辑里的循环就不能缺少&#xff0c;它可以帮助我们实现有规律的重复操作&#xff0c;而…

AIGC-文生视频

stable diffusion&#xff1a; stable diffusion原理解读通俗易懂&#xff0c;史诗级万字爆肝长文&#xff0c;喂到你嘴里 - 知乎个人网站一、前言&#xff08;可跳过&#xff09;hello&#xff0c;大家好我是 Tian-Feng&#xff0c;今天介绍一些stable diffusion的原理&#…

【Python基础】字符集与字符编码

先行了解的知识&#xff1a; 1. 编码和解码 计算机内存储的信息都是二进制表示。 我们看到的英文&#xff0c;数字&#xff0c;汉字等在计算机内如何表示&#xff0c;那就需要编码 计算机内存储的信息需要解析出来&#xff0c;那就是解码 2.字符集与分类 什么是字符集&#xf…

力扣5.最长回文子串

题目描述 思路 1.能够反复利用已判断好的回文子串 2.当子串s[i1,j-1]是回文子串时&#xff0c;只要s[i]s[j]&#xff0c;那么s[i,j]也会是回文子串 3.用好动态规划&#xff0c;具体解释在代码注释里 代码 class Solution {public String longestPalindrome(String s) {int…

Redis分布式锁学习总结

⭐️ 前言 想必大家都有过并发编程的经验&#xff0c;在一个单体应用中&#xff0c;可以通过java提供的各种锁机制来控制多线程对于单体应用中同一资源的并发访问&#xff1b;那么在分布式场景下&#xff0c;想要控制多个应用对于同一外部资源的并发访问&#xff0c;就要用到分…

利用python实现文件压缩打包的功能

主要是利用了zipfile实现文件压缩打包&#xff0c;简单实例代码如下&#xff1a; import zipfilewith zipfile.ZipFile("archive.zip",w) as zipf:zipf.write("config.ini")zipf.write("test.py") 其中的模式 w表示如果没有该文件则创建该文件…