计算机基础--->数据结构(1)【图的存储和遍历】

news2025/1/9 11:27:47

文章目录

    • 图的存储
    • 图的搜索(无向无权图)
      • 代码演示

图中包含 顶点、边、度,无向图,有向图,无权图,带权图,其中 表示一个顶点包含多少条边,有出度和入度。

图的存储

  • 邻接矩阵

在这里插入图片描述
代码

/**
 * 图的存储结构
 */
public class Graph {

    // 存储顶点
    private ArrayList<String> vertextList;
    // 需要一个二维数组,存储邻接矩阵
    private int[][] edges;
    // 边的个数
    private int numofEdges;
    // 顶点是否已经访问
    private static boolean[] isVisited;

    // n: 边数
    public Graph(int n) {
        vertextList = new ArrayList<>();
        isVisited = new boolean[n];
        edges = new int[n][n];
        numofEdges = 0;
    }

    // 如何添加顶点
    public void insertVertex(String vertex) {
        vertextList.add(vertex);
    }

    /**
     * @param e1
     * @param e2
     * @param weight 边的权值,0代表无边,1代表有边
     */
    public void insertEdges(int e1, int e2, int weight) {
        edges[e1][e2] = weight;
        edges[e2][e1] = weight;
        numofEdges++;
    }

    // 返回节点的个数
    public int getNumofVertex() {
        return vertextList.size();
    }

    // 返回边的个数
    public int getNumofEdges() {
        return numofEdges;
    }

    // 获取临界矩阵对应下标的值
    public int getWeight(int e1, int e2) {
        return edges[e1][e2];
    }

    public String getValueByIndex(int index) {
        return vertextList.get(index);
    }

    // 显示矩阵
    public void showGraph() {
        for (int[] e : edges) {
            System.out.println(Arrays.toString(e));
        }
    }

    // 获取第一个邻接点的下标
    public int getFirstNeighbor(int index) {
        for (int i = 0; i < vertextList.size(); i++) {
            if (edges[index][i] > 0) {
                return i;
            }
        }
        return -1;
    }

    // 通过前一个邻接节点的下标获取下一个邻接节点
    public int getNextNeighbor(int e1, int e2) {
        for (int i = e2 + 1; i < vertextList.size(); i++) {
            if (edges[e1][i] > 0) {
                return i;
            }
        }
        return -1;
    }

相当于一个二维数组,当存储无向图时,两点相连标1,当存储有向图时,横向的是起始点指向相应的点标1。优点是简单高效,缺点是浪费空间

  • 邻接表存储

在这里插入图片描述

无向图是将每个点相连的点用链表存储,无向图邻接表中元素的边是途中边的2倍。

有向图是将每个点指向的点列出来,有向图邻接表中元素的边与图中的边数相同。

图的搜索(无向无权图)

  • 广度优先搜索(使用队列)

过程:

  1. 初始状态,将原定点放入队列
  2. 取出堆首节点(搜索数据),将这个堆首的后继未访问过的顶点放入队列
  3. 重复2步骤
  4. 最终取出所有的对首节点即为搜索数据

代码

 // 广度优先遍历
    public void bfs(boolean[] ifVisited, int index) {
        // 获得头节点
        int headIndex;
        // 邻接点w
        int w;
        // 用队列模拟操作
        LinkedList<Integer> linkedList = new LinkedList<>();

        // 输出当前节点
        System.out.print(getValueByIndex(index) + "->");

        // 节点访问,进行标记
        isVisited[index] = true;
        // 队列中存入下标
        linkedList.add(index);

        // 当队列非空,就继续执行,负责结束。
        while (!linkedList.isEmpty()) {

            // 获取头节点下标
            headIndex = linkedList.removeFirst();
            // 获取邻接点下标
            w = getFirstNeighbor(headIndex);

            while (w > 0) {
                // 若邻接点w存在,访问邻接点,并标记已访问,节点w入队列
                while (!isVisited[w]) {
                    System.out.print(getValueByIndex(w) + "->");
                    isVisited[w] = true;
                    linkedList.add(w);
                }
                // 下一个邻接节点要出来
                w = getNextNeighbor(headIndex, w);
            }

        }
    }

    public void bfs() {
        isVisited = new boolean[getNumofVertex()];
        for (int i = 0; i < getNumofVertex(); i++) {
            if (!isVisited[i]) {
                bfs(isVisited, i);
            }
        }
    }
  • 深度优先搜索(使用栈)

过程:

  1. 初始状态,将原定点放入栈中
  2. 取出栈顶元素(搜索栈),将这个栈顶的后继未访问过的顶点放入栈
  3. 重复2步骤
  4. 最终取出所有的栈顶元素即为搜索数据

代码

   // 深度优先遍历
    public void dfs(boolean[] isVisited, int index) {
        System.out.print(getValueByIndex(index) + "->");
        // 如果为true,表示当前节点已经访问
        isVisited[index] = true;
        // 获取邻接点的下标
        int w = getFirstNeighbor(index);
        while (w != -1) {
            if (!isVisited[w]) {
                dfs(isVisited, w);
            }
            w = getNextNeighbor(index, w);
        }
    }

    // 深度遍历时需要回溯
    public void dfs() {
        for (int i = 0; i < getNumofVertex(); i++) {
            if (!isVisited[i]) {
                dfs(isVisited, i);
            }
        }
    }

在这里插入图片描述

代码演示


import java.util.*;

/**
 * 图的存储结构
 */
public class Graph {

    // 存储顶点
    private ArrayList<String> vertextList;
    // 需要一个二维数组,存储邻接矩阵
    private int[][] edges;
    // 边的个数
    private int numofEdges;
    // 顶点是否已经访问
    private static boolean[] isVisited;

    // n: 边数
    public Graph(int n) {
        vertextList = new ArrayList<>();
        isVisited = new boolean[n];
        edges = new int[n][n];
        numofEdges = 0;
    }

    // 如何添加顶点
    public void insertVertex(String vertex) {
        vertextList.add(vertex);
    }

    /**
     * @param e1
     * @param e2
     * @param weight 边的权值,0代表无边,1代表有边
     */
    public void insertEdges(int e1, int e2, int weight) {
        edges[e1][e2] = weight;
        edges[e2][e1] = weight;
        numofEdges++;
    }

    // 返回节点的个数
    public int getNumofVertex() {
        return vertextList.size();
    }

    // 返回边的个数
    public int getNumofEdges() {
        return numofEdges;
    }

    // 获取临界矩阵对应下标的值
    public int getWeight(int e1, int e2) {
        return edges[e1][e2];
    }

    public String getValueByIndex(int index) {
        return vertextList.get(index);
    }

    // 显示矩阵
    public void showGraph() {
        for (int[] e : edges) {
            System.out.println(Arrays.toString(e));
        }
    }

    // 获取第一个邻接点的下标
    public int getFirstNeighbor(int index) {
        for (int i = 0; i < vertextList.size(); i++) {
            if (edges[index][i] > 0) {
                return i;
            }
        }
        return -1;
    }

    // 通过前一个邻接节点的下标获取下一个邻接节点
    public int getNextNeighbor(int e1, int e2) {
        for (int i = e2 + 1; i < vertextList.size(); i++) {
            if (edges[e1][i] > 0) {
                return i;
            }
        }
        return -1;
    }

    // 深度优先遍历
    public void dfs(boolean[] isVisited, int index) {
        System.out.print(getValueByIndex(index) + "->");
        // 如果为true,表示当前节点已经访问
        isVisited[index] = true;
        // 获取邻接点的下标
        int w = getFirstNeighbor(index);
        while (w != -1) {
            if (!isVisited[w]) {
                dfs(isVisited, w);
            }
            w = getNextNeighbor(index, w);
        }
    }

    // 深度遍历时需要回溯
    public void dfs() {
        for (int i = 0; i < getNumofVertex(); i++) {
            if (!isVisited[i]) {
                dfs(isVisited, i);
            }
        }
    }

    // 广度优先遍历
    public void bfs(boolean[] ifVisited, int index) {
        // 获得头节点
        int headIndex;
        // 邻接点w
        int w;
        // 用队列模拟操作
        LinkedList<Integer> linkedList = new LinkedList<>();

        // 输出当前节点
        System.out.print(getValueByIndex(index) + "->");

        // 节点访问,进行标记
        isVisited[index] = true;
        // 队列中存入下标
        linkedList.add(index);

        // 当队列非空,就继续执行,负责结束。
        while (!linkedList.isEmpty()) {

            // 获取头节点下标
            headIndex = linkedList.removeFirst();
            // 获取邻接点下标
            w = getFirstNeighbor(headIndex);

            while (w > 0) {
                // 若邻接点w存在,访问邻接点,并标记已访问,节点w入队列
                while (!isVisited[w]) {
                    System.out.print(getValueByIndex(w) + "->");
                    isVisited[w] = true;
                    linkedList.add(w);
                }
                // 下一个邻接节点要出来
                w = getNextNeighbor(headIndex, w);
            }

        }
    }

    public void bfs() {
        isVisited = new boolean[getNumofVertex()];
        for (int i = 0; i < getNumofVertex(); i++) {
            if (!isVisited[i]) {
                bfs(isVisited, i);
            }
        }
    }


    public static void main(String[] args) {

        int n = 4;
        Graph graph = new Graph(n);

        // 添加顶点
        graph.insertVertex("A");
        graph.insertVertex("B");
        graph.insertVertex("C");
        graph.insertVertex("D");

        // 添加边
        graph.insertEdges(1, 0, 1);
        graph.insertEdges(1, 3, 1);
        graph.insertEdges(3, 0, 1);
        graph.insertEdges(3, 2, 1);

        graph.showGraph();
        System.out.println("深度优先遍历");
        graph.dfs();
        System.out.println("\n广度优先遍历");
        graph.bfs();
    }
}

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

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

相关文章

【LeetCode】13,罗马数字转整数。 难度等级:简单。知识点:map和unordered_map的区别

文章目录 一、题目二、初级解法&#xff1a;顺序遍历字符串我的解法&#xff08;语法平平无奇&#xff09;语法接近 三、精妙解法&#xff1a;逆序遍历字符串四、知识点&#xff1a;map和unordered_map的区别 LeetCode 第13题&#xff0c;罗马数字转整数&#xff1b;难度等级&a…

大厂视频面试,因为截屏作废

大厂视频面试现在这么严格了么&#xff1f;无意间按到截屏直接显示面试作废&#xff0c;好在最后和HR解释了下&#xff0c;再约时间重新面。 作为一个面试过3、4家大厂&#xff0c;现在在鹅厂工作的过来人来说&#xff0c;上面遇到的这个问题是AI面&#xff0c;不用太担心&…

React项目总结:上一步的终点,下一步的起点

项目简介 本人利用 react18.2 json-server 做了一个后台管理系统。 包含&#xff1a; 用户管理权限管理站内信审核管理站内信发布管理 等内容。 其中涉及到react-router V6.0的使用以及一些权限控制等内容。 更多精彩内容&#xff0c;请微信搜索“前端爱好者“&#xff…

Makefile基础教程(函数的使用)

文章目录 前言一、自定义函数1.使用示例2.注意事项 二、预定义函数1.call函数2.abspath函数 总结 前言 在Makefile中也是存在函数的&#xff0c;在 Makefile 中&#xff0c;可以使用函数调用来处理变量、字符串和路径等操作。那么下面就来看看是如何在makefile中使用函数的吧。…

计算机网络基础(四)—— 什么是TCP/IP协议?是两种网络协议?

文章目录 01 | &#x1f353; 概念 \color{red}{概念} 概念&#x1f353;02 | &#x1f34a; T C P / I P 分层模型 \color{orange}{TCP/IP分层模型} TCP/IP分层模型&#x1f34a;03 | &#x1f34b; 数据链路层协议 \color{yellow}{数据链路层协议} 数据链路层协议&#x1f34…

景区剧本杀小程序

景区剧本杀具有以下几个方面的前景&#xff1a; 景区旅游升级&#xff1a;随着人们对于景区旅游体验的多样化需求增加&#xff0c;景区剧本杀作为一种互动性强、参与感强的旅游体验项目&#xff0c;将会得到越来越多游客的喜爱和关注。 移动互联网应用&#xff1a;景区…

C++ 有元 内部类 匿名对象

有元 使用有元就可以突破封装&#xff0c;可以直接对类当中 私有的 成员 成员函数等等进行访问&#xff0c;在某一次上提供了遍历&#xff0c;但是这增大的 耦合性&#xff0c;破坏了封装&#xff0c;所以建议有元不要多用。 所谓耦合性就是 &#xff0c;某两个 东西的 关系&a…

程序员开发Linux常用命令

本文对程序开发过程中常用的Linux命令进行总结&#xff0c;随时进行补充&#xff0c;属于科普篇&#xff0c;希望对大家有所帮助 file命令 该命令用于查看文件的基本信息&#xff0c;比如编码格式&#xff0c;文件类型等信息&#xff0c;对于可执行程序或者动态链接库文件&am…

【Python入门知识】类和对象,要想学的好基础得打好

前言 嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! Python 类/对象 Python 是一种面向对象的编程语言。 Python 中的几乎所有东西都是对象&#xff0c;拥有属性和方法。 类&#xff08;Class&#xff09;类似对象构造函数&#xff0c;或者是用于创建对象的“蓝图”。 创建…

Java 判空的常见方法

一、 对象判空 if (obj ! null) {// 进行对象非空判断 }Object obj null; // 或者 obj new Object(); if (obj null) {// 对象为空 }另外&#xff0c;Guava 库还提供了一个更方便的方法&#xff0c;使用方式如下&#xff1a; import com.google.common.base.Objects;if (O…

SOLIDWORKS 30个实用小技巧

很多人在学习SolidWorks时&#xff0c;会有很多疑问&#xff0c;都不知道如何解答&#xff0c;所以走了很多弯路。今天&#xff0c;我们就来讲讲在学习SolidWorks中的那些小技巧吧&#xff01; 1、SOLIDWORKS技巧之按“空格键&#xff1a;”弹出快捷菜单双击某一视图&#xff0…

webSocket介绍及项目实战【在线聊天系统】

文章目录 一&#xff1a;消息推送常用方式介绍1.1 轮询&#xff1a;浏览器以指定的时间间隔向服务器发出HTTP请求&#xff0c;服务器实时返回数据给浏览器1.2 长轮询&#xff1a;浏览器发出ajax请求&#xff0c;服务器端接收到请求后&#xff0c;会阻塞请求直到有数据或者超时才…

学习之-Spring Cache缓存框架应用本地缓存

此文章用于个人学习记录&#xff0c;原文地址&#xff1a;https://zhuanlan.zhihu.com/p/452315531 如果想了解springCache与redis的交互请看其他文章 缓存是web项目不可或缺的一部分&#xff0c;通过缓存能够降低服务器数据库压力&#xff0c;提高服务器的稳定性及响应速度。…

Rust + 嵌入式:强力开发组合

Rust 的由来 Rust 编程语言的灵感诞生于一次意外。2006年&#xff0c;当 Graydon Hoare 回到位于温哥华的公寓时&#xff0c;发现电梯又因为软件崩溃出了故障。住在 21 楼的他无奈爬楼时&#xff0c;不禁心想&#xff0c;“我们搞计算机的&#xff0c;怎么连个能正常运行的电梯…

接口测试要会什么技能?测试老鸟总结,从接口到接口自动化全面盲扫...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Python自动化测试&…

YOLOv5改进:引入DenseNet思想打造密集连接模块,彻底提升目标检测性能

目录 一、密集连接模块的介绍1、密集连接的概念2、密集连接与残差连接的对比3、DenseNet的结构 二、 YOLOv5中引入密集连接模块的原因1、密集连接模块对于目标检测的优势2、密集连接模块对目标检测性能的影响 三、 YOLOv5中密集连接模块的具体实现1、使用DenseNet的基本单元Den…

怎样才能尽快从开发岗转到产品经理岗位?

越来越多的开发同学随着工作年限的增长都会产生类似的想法。 当然&#xff0c;背后的原因也是多种多样&#xff0c;像薪资上的限制、行业前景的担忧等等&#xff0c;很多同学则踌躇在原地不敢转产品岗位&#xff0c;而有的同学则是通过各种不同方式顺利转岗到产品经理岗位&…

树莓派从源码构建安装Git最新版

1、查看Git版本 首先我们通过SSH客户端连接树莓派,在树莓派中通过查看 Git 版本信息&#xff0c;我们只能看到最高版本显示为 2.30.2&#xff0c;并且通过apt安装也无法将Git更新到最新版。 git --version sudo apt upgrade git那么我们只能通过从源代码来构建安装Git了&…

单片机复习题第二章

1.在AT89S52单片机中&#xff0c;如果采用6MHz晶振&#xff0c;一个机器周期为2us。 时钟周期的振荡频率为fosc&#xff0c;则时钟周期T1/fosc。 一个机器周期包括12个时钟周期。 2.内部RAM中&#xff0c;位地址为40H&#xff0c;88H的位&#xff0c;该位所在字节的字节地址分别…

scrollIntoView 的使用

描述 将调用此方法的元素滚动到浏览器窗口的可见区域。 scrollIntoView 官方文档 用法 element.scrollIntoView() 用法同 element.scrollIntoView(true) element.scrollIntoView(alignToTop) true 表示 element 元素顶部与可见区域的顶部对齐&#xff0c;默认值false 表示 el…