当前位置: 首页 > article >正文

数据结构第一弹-图

大家好,今天和大家一起学习一下数据结构中的图~

图(Graph)是一种非线性的数据结构,它由节点(也称为顶点,Vertices)和边(Edges)组成。图可以用来表示各种关系网络,如社交网络、交通网络、计算机网络等。图的灵活性使得它在许多领域中都有广泛的应用,包括但不限于算法设计、数据库系统、编译器优化以及机器学习。

1. 图的基本概念

1.1 节点与边

  • 节点(Vertex/Node): 图中的基本单位,代表一个实体。
  • (Edge): 连接两个节点的连线,表示这两个节点之间的某种关系。

1.2 图的类型

  • 无向图(Undirected Graph): 边没有方向性,如果存在一条从A到B的边,则也存在一条从B到A的边。
  • 有向图(Directed Graph): 边具有方向性,一条从A到B的边并不意味着存在一条从B到A的边。
  • 加权图(Weighted Graph): 每条边都关联有一个权重值,用于表示边上的成本或距离。
  • 连通图(Connected Graph): 在无向图中,任意两个顶点之间都存在路径;在有向图中,至少存在一个顶点能够到达其他所有顶点。
  • 完全图(Complete Graph): 每对不同的顶点之间都恰好有一条边相连。

1.3 常见术语

  • (Degree): 一个节点连接的边的数量。
  • 入度(In-degree): 在有向图中,指向该节点的边数。
  • 出度(Out-degree): 在有向图中,从该节点出发的边数。
  • 路径(Path): 一系列相邻节点组成的序列。
  • (Cycle): 一条起始节点和结束节点相同的路径。
  • 最短路径(Shortest Path): 两节点间所有可能路径中最短的一条。

2. 图的存储方式

图可以通过多种方式存储,其中最常见的两种是邻接矩阵和邻接表。

2.1 邻接矩阵

邻接矩阵是一个二维数组,用于表示图中各节点间的连接情况。如果A[i][j]为1,则表示节点i和节点j之间存在一条边;如果是0,则表示不存在边。对于加权图,可以用具体权重代替1。

public class AdjacencyMatrix {

    private final int V; // 顶点数量
    private int[][] adjMatrix; // 邻接矩阵
    public AdjacencyMatrix(int v) {
        V = v;
        adjMatrix = new int[V][V];
    }

    public void addEdge(int u, int v, int weight) {
        adjMatrix[u][v] = weight;
        if (u != v) { // 对于无向图
            adjMatrix[v][u] = weight;
        }
    }

    public void printMatrix() {
        for (int i = 0; i < V; i++) {
            for (int j = 0; j < V; j++) {
                System.out.print(adjMatrix[i][j] + " ");
            }
            System.out.println();
        }
    }
}

2.2 邻接表

邻接表使用链表或数组来存储每个节点的所有邻居节点。这种方式在稀疏图中更加高效。


import java.util.ArrayList;
import java.util.List;

public class AdjacencyList {
    private final int V; // 顶点数量
    private List<List<Integer>> adjList; // 邻接表

    public AdjacencyList(int v) {
        V = v;
        adjList = new ArrayList<>(V);
        for (int i = 0; i < V; ++i) {
            adjList.add(new ArrayList<>());
        }
    }

    public void addEdge(int u, int v) {
        adjList.get(u).add(v);
        if (u != v) { // 对于无向图
            adjList.get(v).add(u);
        }
    }


    public void printList() {
        for (int i = 0; i < V; i++) {
            System.out.print("Adjacency list of vertex " + i + ":");
            for (int j : adjList.get(i)) {
                System.out.print(" -> " + j);
            }
            System.out.println();
        }
    }
}

3. 图的遍历算法

图的遍历是指访问图中所有顶点的过程。常用的遍历方法有两种:深度优先搜索(DFS)和广度优先搜索(BFS)。

3.1 深度优先搜索(DFS)

DFS从一个给定的起点开始,尽可能深地探索每个分支,直到无法继续为止,然后回溯并尝试另一条路径。

import java.util.*;

public class DFS {
    private boolean[] visited;

    public DFS(int vertices) {
        visited = new boolean[vertices];
    }

    public void dfs(AdjacencyList graph, int start) {
        Stack<Integer> stack = new Stack<>();
        stack.push(start);

        while (!stack.isEmpty()) {
            int s = stack.pop();

            if (!visited[s]) {
                System.out.print(s + " ");
                visited[s] = true;
            }

            for (int n : graph.adjList.get(s)) {
                if (!visited[n]) {
                    stack.push(n);
                }
            }
        }
    }
}

3.2 广度优先搜索(BFS)

BFS则从一个给定的起点开始,逐层向外扩展,先访问离起点最近的所有节点,然后再访问更远一层的节点。

import java.util.*;

public class BFS {
    private boolean[] visited;
    public BFS(int vertices) {
        visited = new boolean[vertices];
    }

    public void bfs(AdjacencyList graph, int start) {
        Queue<Integer> queue = new LinkedList<>();
        queue.add(start);
        visited[start] = true;

        while (!queue.isEmpty()) {
            int s = queue.poll();
            System.out.print(s + " ");

            for (int n : graph.adjList.get(s)) {
                if (!visited[n]) {
                    queue.add(n);
                    visited[n] = true;
                }
            }
        }
    }
}

4. 图的应用实例

4.1 社交网络分析

社交网络可以看作是有向图,用户作为节点,好友关系作为边。通过图论算法,我们可以计算用户的影响力、社区检测等。

4.2 路径规划

在地图应用中,城市作为节点,道路作为边。利用Dijkstra算法或A*算法,可以找到两点之间的最短路径。

import java.util.*;

public class Dijkstra {
    private static final int INF = Integer.MAX_VALUE;

    public static int[] dijkstra(AdjacencyList graph, int start) {
        int n = graph.V;
        int[] dist = new int[n];
        Arrays.fill(dist, INF);
        dist[start] = 0;
        PriorityQueue<int[]> pq = new PriorityQueue<>(Comparator.comparingInt(a -> a[1]));
        pq.offer(new int[]{start, 0});

        while (!pq.isEmpty()) {
            int[] current = pq.poll();
            int u = current[0];
            if (dist[u] < current[1]) continue;
            for (int v : graph.adjList.get(u)) {
                if (dist[u] + 1 < dist[v]) { // 假设所有边的权重都是1
                    dist[v] = dist[u] + 1;
                    pq.offer(new int[]{v, dist[v]});
                }
            }
        }

        return dist;
    }
}

4.3 数据库索引

在数据库中,图可以用来表示表之间的关系。例如,在关系型数据库中,外键约束可以形成一张图,帮助优化查询性能。

4.4 编译器优化

编译器在进行代码优化时,可以将程序转换成控制流图(CFG),通过对图的操作来识别冗余代码、循环不变量等,从而提高程序效率。

图作为一种强大的数据结构,提供了丰富的表达能力和灵活的处理机制。无论是简单的关系表示还是复杂的网络分析,图都能提供有效的解决方案,欢迎大家一起在评论区讨论~


http://www.kler.cn/a/430390.html

相关文章:

  • Unity的四种数据持久化方式
  • 嵌入式技术之Linux(Ubuntu) 一
  • 基于SpringBoot实现的保障性住房管理系统
  • Ollama + FastGPT搭建本地私有企业级AI知识库 (Linux)
  • Redis中的主从/Redis八股
  • 如何快速上手一个鸿蒙工程
  • sql中的运算符
  • 数据结构题库11
  • 网络安全法-法律责任
  • 酷柚易汛进销存系统PHP+Uniapp
  • 上下料机器人,CNC 加工行业的变革性力量
  • 电动车多动力源的扭矩协调:提升能源效率与车辆性能的关键
  • 如何优化 JavaScript 中的 DOM 操作?
  • Linux: docker: 怎么修改 proc下的文件内容?
  • 一次“okhttp访问间隔60秒,提示unexpected end of stream“的问题排查过程
  • 【开源】A065—基于SpringBoot的库存管理系统的设计与实现
  • 目标跟踪领域经典论文解析
  • WPF+LibVLC开发播放器-音量控制和倍速控制
  • AKE 安全模型:CK, CK+, eCK
  • 火山引擎数据飞轮最新活动:结合大模型能力,探索金融行业数智化落地新可能
  • vue实现导出excel表,调整图片大小
  • ios逆向某新闻 md5+aes
  • Oracle清除水位
  • 5G Multicast/Broadcast Services(MBS) (八) MBS多播DRX
  • Reclone映射webdav到本地(Linux)
  • 《探索形象克隆:科技与未来的奇妙融合》