关于图结构这一篇就够了

图(Graph)是一种非线性数据结构,由节点(或顶点,Vertex)和边(Edge)组成。边可能带有权重,表示两个节点之间的某种关系或距离。图结构广泛应用于计算机科学中,如网络路由、社交网络分析、数据结构优化等。

基本概念

  • 节点(Vertex):图中的基本元素,通常表示为一个点或对象。
  • 边(Edge):连接两个节点的线段或弧线,表示两个节点之间的关系。
  • 无向图(Undirected Graph):边没有方向的图。
  • 有向图(Directed Graph):边有方向的图,通常表示为箭头。
  • 权重(Weight):边可能带有的数值,表示两个节点之间的某种度量或距离。
  • 邻接节点(Adjacent Vertex):通过一个边与给定节点相连的所有节点。
  • 度(Degree):无向图中一个节点的邻接节点的数量;有向图中分为入度(指向该节点的边的数量)和出度(从该节点出发的边的数量)。

特点

  • 非线性:节点之间的关系不是线性的,可以任意连接。
  • 灵活性:可以表示复杂的关系网络。
  • 多样性:包括无向图、有向图、加权图等多种类型。

代码实现

下面是一个使用Java实现的简单无向图,包括准备数据、创建图、清空图、显示图和遍历图(深度优先搜索DFS和广度优先搜索BFS)的基本操作。

首先,我们定义一个Vertex类来表示图的顶点,以及一个Graph类来表示图本身:

下面是一个使用Java实现的简单无向图,包括准备数据、创建图、清空图、显示图和遍历图(深度优先搜索DFS和广度优先搜索BFS)的基本操作。

首先,我们定义一个Vertex类来表示图的顶点,以及一个Graph类来表示图本身:

import java.util.*;
class Vertex {
String label;
Vertex(String label) {
this.label = label;
}
@Override
public String toString() {
return label;
}
}
class Graph {
private Map<Vertex, List<Vertex>> adjList;
Graph() {
adjList = new HashMap<>();
}
void addVertex(Vertex v) {
adjList.put(v, new ArrayList<>());
}
void addEdge(Vertex v1, Vertex v2) {
adjList.get(v1).add(v2);
adjList.get(v2).add(v1); // 无向图,所以两个方向都要添加
}
void removeVertex(Vertex v) {
for (List<Vertex> neighbors : adjList.values()) {
neighbors.remove(v);
}
adjList.remove(v);
}
void clearGraph() {
adjList.clear();
}
void displayGraph() {
for (Map.Entry<Vertex, List<Vertex>> entry : adjList.entrySet()) {
System.out.println(entry.getKey() + " -> " + entry.getValue());
}
}
// 深度优先搜索
void dfs(Vertex v, boolean[] visited) {
visited[v.hashCode()] = true;
System.out.print(v + " ");
for (Vertex neighbor : adjList.get(v)) {
if (!visited[neighbor.hashCode()]) {
dfs(neighbor, visited);
}
}
}
void dfs(Vertex startVertex) {
boolean[] visited = new boolean[100]; // 假设图的大小不会超过100个顶点
Arrays.fill(visited, false);
dfs(startVertex, visited);
}
// 广度优先搜索
void bfs(Vertex startVertex) {
Queue<Vertex> queue = new LinkedList<>();
Set<Vertex> visited = new HashSet<>();
queue.add(startVertex);
visited.add(startVertex);
while (!queue.isEmpty()) {
Vertex v = queue.poll();
System.out.print(v + " ");
for (Vertex neighbor : adjList.get(v)) {
if (!visited.contains(neighbor)) {
queue.add(neighbor);
visited.add(neighbor);
}
}
}
}
public static void main(String[] args) {
Graph g = new Graph();
// 准备数据
Vertex A = new Vertex("A");
Vertex B = new Vertex("B");
Vertex C = new Vertex("C");
Vertex D = new Vertex("D");
Vertex E = new Vertex("E");
Vertex F = new Vertex("F");
// 创建图
g.addVertex(A);
g.addVertex(B);
g.addVertex(C);
g.addVertex(D);
g.addVertex(E);
g.addVertex(F);
g.addEdge(A, B);
g.addEdge(A, C);
g.addEdge(B, D);
g.addEdge(C, D);
g.addEdge(D, E);
g.addEdge(E, F);
// 显示图
System.out.println("Graph:");
g.displayGraph();
// 遍历图(深度优先搜索)
System.out.println("\nDFS from A:");
g.dfs(A);
// 遍历图(广度优先搜索)
System.out.println("\nBFS from A:");
g.bfs(A);
// 清空图
g.clearGraph();
// 显示图(应为空)
System.out.println("\nGraph after clearing:");
g.displayGraph();
}
}

注意,在上面的示例中,我们使用了hashCode()方法和数组来跟踪顶点是否已经被访问过,这可能在某些情况下(例如,当多个不同对象具有相同的哈希码时)导致问题。为了避免这种情况,我们可以使用一个HashSet来跟踪访问过的顶点,因为HashSet直接依赖于对象的equals()hashCode()方法,但它在添加元素时会检查元素是否已存在。

然而,在DFS的实现中,如果我们使用HashSet来跟踪访问过的顶点,我们就不需要为整个图预先分配一个布尔数组。

另外,如果图可能非常大,我们不能预先假设一个固定大小的布尔数组来跟踪访问过的顶点。下面是一个改进的DFS实现,它使用HashSet来跟踪访问过的顶点:

// ... 省略其他部分 ...
// 深度优先搜索
void dfs(Vertex startVertex) {
Set<Vertex> visited = new HashSet<>();
Stack<Vertex> stack = new Stack<>();
stack.push(startVertex);
while (!stack.isEmpty()) {
Vertex v = stack.pop();
if (!visited.contains(v)) {
visited.add(v);
System.out.print(v + " ");
for (Vertex neighbor : adjList.get(v)) {
if (!visited.contains(neighbor)) {
stack.push(neighbor);
}
}
}
}
}
// ... 省略其他部分 ...

在这个改进的实现中,我们使用了一个Stack来模拟深度优先搜索的递归过程,同时用一个HashSet来跟踪访问过的顶点。当遍历到新的邻居时,如果它还没有被访问过,我们就将其添加到栈中以便后续处理。

这个实现更加健壮,因为它不依赖于图的顶点数量,并且不依赖于顶点的hashCode()方法的实现。它可以处理任意大小的图和具有任意hashCode()方法的顶点。

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

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

相关文章

渗透之sql盲注(时间/boolean盲注)

sql盲注&#xff1a;sql盲注意思是我们并不能在web页面中看到具体的信息&#xff0c;我们只能通过输入的语句的真假来判断。从而拿到我们想要的信息。 我们通常使用ascii值来进行盲注。 目录 手动注入&#xff1a; 时间盲注&#xff1a; 布尔盲注&#xff1a; python脚本注…

2024阿里云ctf-web-chain17学习

agent jdk17依赖有h2思路清晰打jdbc attack <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- https://mvnrepository.com/artifact/com.aliba…

【Spark】Spark分布式环境安装(二)

Spark分布式环境安装 将spark-3.5.0-bin-hadoop3.tgz 文件上传到 Linux 并解压缩&#xff0c;放置在指定位置&#xff0c;路径中不要包含中文或空格&#xff0c;解压缩操作&#xff0c;不再强调。 基于Windows的环境体验 启动脚本 启动界面如图-1所示。 图-1 spark-shell启动…

Vue工程化开发和脚手架Vue CLI

目录 一、介绍 二、使用步骤 1. 全局安装&#xff08;一次&#xff09; 2.查看Vue版本 3.创建项目架子&#xff08;项目名不能使用中文&#xff09; 4.启动项目 一、介绍 Vue CLI是Vue官方提供的一个全局命令工具。可以帮助我们快速创建一个开发的Vue项目的标准化基础架子…

QT:小项目:登录界面 (下一个连接数据库)

一、效果图 登录后&#xff1a; 二、项目工程结构 三、登录界面UI设计 四主界面 四、源码设计 login.h #ifndef LOGIN_H #define LOGIN_H#include <QDialog>namespace Ui { class login; }class login : public QDialog {Q_OBJECTpublic:explicit login(QWidge…

架构每日一学 3:架构师六个生存法则之一:如何找到唯一且正确的架构目标?(二)

本文首发于公众号&#xff1a;腐烂的橘子 上一篇文章中&#xff0c;我们讨论了架构师第一个生存法则&#xff1a;必须有且仅有一个目标。今天我们主要讨论下如何找到这个目标。 确认一个正确目标且要试图逼近它 每一个企业的第一任务首先是活下来&#xff0c;然后再盈利。那么…

unity制作app(3)--gps定位

1.unity中定位Unity之GPS定位&#xff08;高德解析&#xff09;_unity gps定位-CSDN博客 代码需要稍微修改一下&#xff0c;先把脚本绑到一个button上试一试&#xff01; 2.先去高德地图认证&#xff08;app定位&#xff09; 创建应用和 Key-Web服务 API | 高德地图API (ama…

结构体介绍(2)

结构体介绍&#xff08;2&#xff09; 前言一、结构体的内存对齐之深入理解为什么存在内存对齐&#xff1f;修改默认对齐数 二、结构体传参2.1&#xff1a;该怎么传参呢&#xff1f; 三、结构体实现位段3.1什么是位段位段的内存分配位段的跨平台问题 总结 前言 根据之前讲了结…

CMakeLists.txt语法规则:改变行为的变量说明二

一. 简介 前面一篇文章学习了 CMakeLists.txt语法中的 部分常量变量&#xff0c;具体学习提供信息的变量&#xff0c;文章如下&#xff1a; CMakeLists.txt语法规则&#xff1a;提供信息的变量说明一-CSDN博客 CMakeLists.txt语法规则&#xff1a;提供信息的变量说明二-CSD…

【计算机网络原理】万字长文,持续更新...

文章目录&#x1f970; 计算机网络原理1.2 因特网概述1 网络、互联网&#xff08;互连网&#xff09;和因特网2 因特网发展的三个阶段ISP的概念基于ISP的三层结构的因特网 3 因特网的标准化工作4 因特网的组成 1.3 三种交换方式&#xff1a;电路交换、分组交换和报文交换电路交…

ctfshow 框架复现

文章目录 web 466web 467web 468web469web 470web 471web 472web 473web 474web 475web 476 web 466 Laravel5.4版本 &#xff0c;提交数据需要base64编码 代码审计学习—Laravel5.4 - 先知社区 (aliyun.com) 用第二条链子 反序列化格式 /admin/序列化串base64<?php na…

【多模态】29、OCRBench | 为大型多模态模型提供一个 OCR 任务测评基准

文章目录 一、背景二、实验2.1 测评标准和结果2.1.1 文本识别 Text Recognition2.1.2 场景文本中心的视觉问答 Scene Text-Centric VQA2.1.3 文档导向的视觉问答 Document-Oriented VQA2.1.4 关键信息提取 Key Information Extraction2.1.5 手写数学公式识别 Handwritten Mathe…

图片浏览软件-XnView

一、前言 XnView MP / Classic是一款免费的图像查看器&#xff0c;可轻松打开和编辑照片文件。图像查看器支持所有主要的图像格式&#xff08;JPEG&#xff0c;TIFF&#xff0c;PNG&#xff0c;GIF&#xff0c;WEBP&#xff0c;PSD&#xff0c;JPEG2000&#xff0c;OpenEXR&am…

ssm106学生公寓管理系统的设计与实现+jsp

学生公寓管理系统设计与实现 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本学生公寓管理系统就是在这样的大环境下诞生&#xff0c;其可以帮助管理者在短…

matlab例题大全

1.第1章 MATLAB系统环境 1.1 注&#xff1a;plot函数为画图函数。例plot&#xff08;x1,y1,:,x2,y2,*&#xff09;; 1.2 注&#xff1a;root为求根函数。p为方程变量前面系数矩阵。 1.3 注&#xff1a; 2*x3y-1*z 2; 8*x2*y3*z 4; 45*x3*y9*z 23 求&#xff1a;x,y,z的…

python数据分析——大数据和云计算

大数据和云计算 前言一、大数据二、大数据定义三、数据存储单位四、大数据存储技术五、大数据应用技术六、大数据特征七、数据容量八、数据类型的多样性8.1结构化数据8.2半结构化数据8.3非结构化数据 九、获取数据的速度十、可变性十一、真实性十二、复杂性十三、价值十四、云计…

2023下半年软件设计师上午题——冒泡排序

快速排除法&#xff0c;根据冒泡排序特性&#xff0c;每一趟排序都会确实最大/最小值&#xff0c;故升序两趟后&#xff0c;最后两个元素应该是已经排序好的第二大&#xff0c;和最大的元素&#xff0c;所以排除B,D&#xff0c;再因为每次排序都会两两交换&#xff0c;所以排除…

第五十一周:文献阅读+CNN-LSTM-AM

目录 摘要 Abstract 文献阅读&#xff1a;基于CNN-LSTM-AM时空深度学习模型的城市供水预测 现存问题 提出方法 创新点 方法论 CNN-LSTM-AM模型 研究实验 数据集 预处理 评估指标 实验过程 合格性验证 模型实现 总结 摘要 本周阅读的文献《Urban Water Supply …

C#知识|上位机项目主窗体设计思路及流程(实例)

哈喽,你好啊,我是雷工! 昨天练习了登录窗体的设计实现,今天练习上位机项目主窗体的设计实现。 01 主窗体效果展示 02 实现步骤 2.1、添加主窗体 添加窗体,名称:FrmMain.cs 2.2、窗体属性设置 将FrmMain窗体属性FormBorderStyle设置为None,无边框; 将FrmMain窗体属性…

C++进阶 | [2] 多态

摘要&#xff1a;多态的概念&#xff0c;多态的条件&#xff0c;虚函数的重写&#xff0c;抽象类&#xff0c;多态的原理&#xff0c;虚函数与虚函数表&#xff0c;与多态有关的问答题 1. Concept 多态的概念&#xff1a;通俗来说&#xff0c;就是多种形态&#xff0c;具体点就…
最新文章