//图的存储结构:
const int MAXSIZE = 10;
//邻接矩阵
template<class T>
class MGraph {
public:
    MGraph(T a[], int n, int e);
    void DFS(int v);
    void BFS(int v);
private:           //edge为边用来表示无向图,arc为弧用来表示有向图,vertex为顶点
    T vertex[MAXSIZE];
    int arc[MAXSIZE][MAXSIZE];
    int edge[MAXSIZE][MAXSIZE];
    int vNum, arcNum,eNum;
    bool* visited;     //DFS使用
};
template<class T>
MGraph<T>::MGraph(T a[], int v, int e) {     //n为顶点个数,e为边的个数,a[]为各顶点值
    vNum = v;            //将顶点个数,边个数,顶点值放入类中属性
    arcNum = e;
    visited = new bool[v];
    memset(visited, false, sizeof(bool) * v);
    for (int k = 0; k < v; k++) {    
        vertex[k] = a[k];
    }
    for (int k = 0; k < v; k++) {    //先将邻接矩阵初始化
        for (int j = 0; j < v; j++) {
            arc[k][j] = 0;
        }
    }
    for (int k = 0,i=0,j=0; k < e; k++) {  //这里是无向图的邻接矩阵,输入相连的两个顶点
        cin >> i >> j;
        arc[j-1][i-1] = arc[i-1][j-1] = 1;
    }
}

//邻接表,同样以无向图为例,边用arc表示
struct ArcNode {    //边的数据结构,存储边信息和下一条边
    int adjvex;   //相邻的顶点
    ArcNode* nextarc;
};
struct VertexNode {   //顶点的数据结构,存储顶点信息和下一条边
    int vertex;
    ArcNode* firstarc;
};
template<class T>
class ALGraph {
public:
    ALGraph(T a[], int n, int e);
    //~ALGraph();
    void DFS(int v);
    void BFS(int v);
private:
    VertexNode adjlist[MAXSIZE];
    int vNum, arcNum;
    bool* visited;
};
template<class T>
ALGraph<T>::ALGraph(T a[], int v, int e) {
    vNum = v;
    arcNum = e;
    visited = new bool[v];
    memset(visited, false, sizeof(bool) * v);
    for (int i = 0; i < v; i++) {  //对顶点表进行初始化
        adjlist[i].vertex = a[i];
        adjlist[i].firstarc = NULL;
    }
    for (int k = 0,i=0,j=0; k < e; k++) {  //这里是无向图,如果有向图的话,则为顶点i指向j
        cin >> i >> j;               //注意这里的e是每个顶点连接的边的总和,将无向看成有双箭头的有向(实际不适合表示无向图)
        ArcNode* s = new ArcNode;
        s->adjvex = j;                      //这里使用的是头插入法
        s->nextarc = adjlist[i-1].firstarc;
        adjlist[i-1].firstarc = s;
    }
}
//补充:邻接表适合表示有向图,还有一种逆邻接表表示法,与邻接表的出为指向方向相反,逆邻接表类似于入边表

//十字链表,只提供节点的存储结构
struct VertexNodeC {
    int Vertex;
    ArcNode* firstin;
    ArcNode* firstout;
};
struct ArcNodeC {
    int headvex,tailvex;
    ArcNode* hlink;
};

//边集数组,比较常用,使用两个一维数组存储,一个数组存储顶点,一个数组存储起点,终点,权值
struct ArcNodeE {
    int headvex;
    int tailvex;
    int weight;
};

//图的遍历:DFS与BFS
//DFS深度优先搜索,类似于二叉树的前序遍历,先从一条支路到结尾,然后一步步回溯寻找还未访问过的节点
//邻接矩阵和邻接表两种存储结构的dfs
//两种存储结构的dfs比较
//邻接矩阵的时间复杂度 n
//邻接表的时间复杂度 n+e 栈深度 n 空间复杂度 n
template<class T>
void MGraph<T>::DFS(int v) {
    cout << vertex[v];
    visited[v] = true;    //将访问过的顶点标记为访问过
    for (int j = 0; j < vNum; j++) {       //按照邻接矩阵的特性,根据标记来深入搜索
        if (arc[v][j] == 1 && visited[j] == false) {
            DFS(j);
        }
    }
}
template<class T>
void ALGraph<T>::DFS(int v) {  //根据邻接表的连接,从顶点第一个边节点转为顶点节点,并标记是否访问,然后一步步回溯看是否有未访问的相邻节点
    cout << adjlist[v].vertex;
    visited[v] = true;
    ArcNode* p = adjlist[v].firstarc;
    while (p) {
        if (visited[p->adjvex] == false) {
            DFS(p->adjvex);
        }
        p = p->nextarc;
    }
}

//BFS广度优先搜索,类似于树的层序遍历,使用队列逻辑结构
//分别用邻接矩阵和邻接表描述
//邻接矩阵时间复杂度 n
//邻接表时间复杂度 n+e 空间复杂度 n
template<class T>
void MGraph<T>::BFS(int v) {
    int queue[MAXSIZE];
    int f = 0, r = 0;
    cout << vertex[v];
    visited[v] = true;
    queue[++r] = v;
    while (f != r) {      //当头尾相等时队列为空,表示队列为空
        v = queue[++f];
        for (int j = 0; j < vNum; j++) {
            if (arc[v][j] == 1 && visited[j] == false) {
                cout << vertex[j];
                visited[j] = true;
                queue[++r] = j;
            }
        }
    }
}
template<class T>
void ALGraph<T>::BFS(int v) {
    int queue[MAXSIZE];
    int f = 0, r = 0;
    cout << adjlist[v].vertex;
    visited[v] = true;
    queue[++r] = v;
    while (r!=f) {
        v = queue[++f];
        ArcNode *p = adjlist[v].firstarc;
        while (p) {
            int j = p->adjvex;
            if (visited[j] == false) {
                cout << adjlist[j].vertex;
                visited[j] = true;
                queue[++r] = j;
            }
            p = p->nextarc;
        }
    }
}
关于图的4中基本算法放在单独一篇,Prim,Kruskal,Floyd,Dijkstra

图的数据结构的实现与遍历(DFS,BFS)的更多相关文章

  1. 图的遍历[DFS][BFS]

    #include<iostream> #include<iostream> #include<cstring> #include<queue> #inc ...

  2. 列出连通集(DFS及BFS遍历图) -- 数据结构

    题目: 7-1 列出连通集 (30 分) 给定一个有N个顶点和E条边的无向图,请用DFS和BFS分别列出其所有的连通集.假设顶点从0到N−1编号.进行搜索时,假设我们总是从编号最小的顶点出发,按编号递 ...

  3. 图的两种遍历:DFS&BFS

    DFS和BFS在图中的应用: 图连通性判定:路径的存在性:图中是否存在环:求图的最小生成树:求图的关键路径:求图的拓扑排序. DFS:简单的说,先一直往深处走,直到不能再深了,再从另一条路开始往深处走 ...

  4. 图的遍历——DFS和BFS模板(一般的图)

    关于图的遍历,通常有深度优先搜索(DFS)和广度优先搜索(BFS),本文结合一般的图结构(邻接矩阵和邻接表),给出两种遍历算法的模板 1.深度优先搜索(DFS) #include<iostrea ...

  5. 数据结构与算法之PHP用邻接表、邻接矩阵实现图的广度优先遍历(BFS)

    一.基本思想 1)从图中的某个顶点V出发访问并记录: 2)依次访问V的所有邻接顶点: 3)分别从这些邻接点出发,依次访问它们的未被访问过的邻接点,直到图中所有已被访问过的顶点的邻接点都被访问到. 4) ...

  6. 图的深度优先遍历(DFS)和广度优先遍历(BFS)

    body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...

  7. PAT Advanced 1034 Head of a Gang (30) [图的遍历,BFS,DFS,并查集]

    题目 One way that the police finds the head of a gang is to check people's phone calls. If there is a ...

  8. 图的遍历DFS

    图的遍历DFS 与树的深度优先遍历之间的联系 树的深度优先遍历分为:先根,后根 //树的先根遍历 void PreOrder(TreeNode *R){ if(R!=NULL){ visit(R); ...

  9. 算法学习记录-图(DFS BFS)

    图: 目录: 1.概念 2.邻接矩阵(结构,深度/广度优先遍历) 3.邻接表(结构,深度/广度优先遍历) 图的基本概念: 数据元素:顶点 1.有穷非空(必须有顶点) 2.顶点之间为边(可空) 无向图: ...

随机推荐

  1. MYSQL---外键 primary key 作用

    https://www.cnblogs.com/x739400043/p/4732158.html 外键和级联   关于外键写的不错的网址:http://blog.csdn.net/lidaasky/ ...

  2. bootstrap-suggest-plugin input可选可输(表单) 好用的前端插件

    bootstrap-suggest-plugin          DEMO下载 1.准备:页面引入(点击下载) <link rel="stylesheet" href=&q ...

  3. RFID学习

    RFID:射频识别卡有读卡器和电子标签(有源/无源/半有源)组成,工作频率有LF/HF/UHF/MF,低频的通讯距离近(1m)常用来做门禁:高频的通讯距离远可以批量读取. 无源的轻巧/防拆一致性好,在 ...

  4. Java直通车——类与对象篇

    一.面向对象 教师节来临,一群小学生应召出黑板报,该怎样完成呢?对于一个小孩子来说,你告诉他:“我们要确定黑板报主题.明确内容要旨.搜寻具体内容.构思版面布局.画图和写字.安排人员后勤辅助.”他可能会 ...

  5. css简单整理

    style1.css /*统一设置h1.h2.a标签的样式*/ h1,h2,a{ color: aqua; font-size: 50px; } /*如果h1.h2没有上面的单独设置样式那么就会继承b ...

  6. 阿里云短信接口开发实践(Java

    随着互联网的兴起,各行各业的需求都在不断的增加.随着业务的扩大,企业给用户发送短信验证码的业务,也是如火如荼.在这里,calvin给各位开发者推荐阿里云短信平台.原因有二:1.接入较简单,开发成本低 ...

  7. selenium+chrome options

    selenium+chrome options 环境:selenium chrome 1.      selenium + chrome参数配置 1.1.    启动 from selenium im ...

  8. jQuery常用操作(待续)

    1. input清空内容 <1> $("#选择器id").val(""); <2> $("input[name='input框 ...

  9. axios发送post请求[body-parser]--['Content-type': 'application/x-www-form-urlencoded']

    const express = require('express') const axios = require('axios') const bodyParser = require('body-p ...

  10. Python 基础之面向对象之常用魔术方法

    一.__init__魔术属性 触发时机:实例化对象,初始化的时候触发功能:为对象添加成员,用来做初始化的参数:参数不固定,至少一个self参数返回值:无 1.基本用法 #例:class MyClass ...