JS实现图结构封装,使用邻接表实现(广度优先搜索,深度优先搜索)

JS实现图结构使用邻接表实现(广度优先搜索,深度优先搜索)

1.获取到的知识

  • 层序遍历
    • 实现原理:先对每层做标记,如果标记一次进行遍历,遇到已经遍历过的,跳过
  • 巧妙之处:对于图结构的表示中有一种邻接表,类似于哈希表中的链地址法,可以方便的把顶点和每个每条边对应起来,每个边也对应的是一个顶点,这样就可以把图机构抽象成程序储存起来,下图的邻接表可以用链表或者数组都可以实现

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-QRGjgJbV-1601890213224)()]

2. 广度优先搜索和深度优先搜索

2.1 广度优先搜索的巧妙之处 (BFS)

1.广度优先搜索原理主要是利用队列来实现,先把所有的内容都给搜索到,然后再依次的去处理每一个搜索到的内容

2.利用一个对象或者Map来储存是否是已经遍历或者探测过的对象,来避免重复查询

bfs() {
    //0 首先需要一个标识符来确定每个顶点是否已经进入过队列
    let queue = [];
    let vertexesMap = this.isExpload();
    //1. 拿到第一个顶点,加入队列中
    vertexesMap.set(this.vertexes[0], true);
    queue.push(this.vertexes[0]);
    //2. 根据队列不为空循环处理队列的内容
    while (queue.length != 0) {
      //3. 执行队列中的第一个顶点
      //4. 拿到顶点之后,弹出顶点
      let vertexe = queue.shift();
      //5. 对这个顶点执行相应的操作,实际可是传递一个回调函数来进行处理
      console.log(vertexe);
      //6. 循环探测该顶点的边连接的其他的所有顶点。
      let otherVertexes = this.edges.get(vertexe);
      for (let i = 0; i < otherVertexes.length; i++) {
        //7. 对探测的顶点进行判断
        //7.1 如果顶点没有加入,则把该顶点加入队列
        if (!vertexesMap.get(otherVertexes[i])) {
          vertexesMap.set(otherVertexes[i],true);
          queue.push(otherVertexes[i]);
        }
        //7.2 如果顶点已经被加入过队列,则不再加入.
      }

    }
  }

2.2 深度优先搜索

​ 深度优先搜索算法和哈希表的先序、中序、后序、遍历思想差不多一致,只不过多了一个用于储存顶点是否发生改变的状态。

通过储存每个顶点是否已经被探测过,来觉得是否对该顶点进行遍历处理

//深度优先搜索 DeepFirstSearch 
  dfs(){
    //深度搜索原理是通过递归,先把一个分支上的所有内容都遍历完成,然后再去遍历其他的分支
    //1.获取所有状态
    let vertexesMap = this.isExpload();
    this.dfsNode(this.vertexes[0],vertexesMap);
  }
  //遍历处理每个顶点
  dfsNode(vertexe,vertexesMap){
    //1. 需要标识符来判断是否已经处理过。
    
    //1.1 处理传入的顶点,并把顶点切换为已经处理
    console.log(vertexe);
    vertexesMap.set(vertexe,true);
    //2.获取传入顶点的边所对应的顶点;
    let vertexes = this.edges.get(vertexe);
    //3.根据循环遍历获得的顶点
    for(let i=0;i<vertexes.length;i++){
      //4.判断顶点是否已经处理过
      //4.1 如果没有处理过,继续调用该函数,进行处理遍历
      if(  !vertexesMap.get(vertexes[i])  ){
        this.dfsNode(vertexes[i],vertexesMap);
      }
      //4.2 如果处理过则不用管了,
    }

  }

3.完整图结构封装代码

/* 图结构封装 */
class Graph {
  constructor() {
    //需要的属性,图结构表示方法使用:邻接表
    //1.顶点
    this.vertexes = [];
    //2.边
    this.edges = new Map();
  }
  //插入一个顶点
  insertVertexes(v) {
    this.vertexes.push(v);
    this.edges.set(v, []);
  }
  //插入一条边
  insertEdge(v1, v2) {
    this.edges.get(v1).push(v2);
    this.edges.get(v2).push(v1);
  }
  //返回一个Map,包含一个 key(元素) 和 一个val (是否已经被探测过)
  isExpload() {
    let vertexesMap = new Map();
    let vertexes = this.vertexes;
    for (let i = 0; i < vertexes.length; i++) {
      vertexesMap.set(vertexes[i], false);
      let edges = this.edges.get(vertexes[i]);
      for (let j = 0; j < edges.length; j++) {
        vertexesMap.set(edges[j], false);
      }
    }
    return vertexesMap;
  }
  //广度优先搜索 BFS BroundFirstSearch
  bfs() {
    //广度优先搜索原理主要是利用队列来实现,先把所有的内容都给搜索到,然后再依次的去处理每一个搜索到的内容
    //0 首先需要一个标识符来确定每个顶点是否已经进入过队列
    let queue = [];
    let vertexesMap = this.isExpload();
    //1. 拿到第一个顶点,加入队列中
    vertexesMap.set(this.vertexes[0], true);
    queue.push(this.vertexes[0]);
    //2. 根据队列不为空循环处理队列的内容
    while (queue.length != 0) {
      //3. 执行队列中的第一个顶点
      //4. 拿到顶点之后,弹出顶点
      let vertexe = queue.shift();
      //5. 对这个顶点执行相应的操作,实际可是传递一个回调函数来进行处理
      console.log(vertexe);
      //6. 循环探测该顶点的边连接的其他的所有顶点。
      let otherVertexes = this.edges.get(vertexe);
      for (let i = 0; i < otherVertexes.length; i++) {
        //7. 对探测的顶点进行判断
        //7.1 如果顶点没有加入,则把该顶点加入队列
        if (!vertexesMap.get(otherVertexes[i])) {
          vertexesMap.set(otherVertexes[i],true);
          queue.push(otherVertexes[i]);
        }
        //7.2 如果顶点已经被加入过队列,则不再加入.
      }

    }
  }
  //深度优先搜索 DeepFirstSearch 
  dfs(){
    //深度搜索原理是通过递归,先把一个分支上的所有内容都遍历完成,然后再去遍历其他的分支
    //1.获取所有状态
    let vertexesMap = this.isExpload();
    this.dfsNode(this.vertexes[0],vertexesMap);
  }
  //遍历处理每个顶点
  dfsNode(vertexe,vertexesMap){
    //1. 需要标识符来判断是否已经处理过。
    
    //1.1 处理传入的顶点,并把顶点切换为已经处理
    console.log(vertexe);
    vertexesMap.set(vertexe,true);
    //2.获取传入顶点的边所对应的顶点;
    let vertexes = this.edges.get(vertexe);
    //3.根据循环遍历获得的顶点
    for(let i=0;i<vertexes.length;i++){
      //4.判断顶点是否已经处理过
      //4.1 如果没有处理过,继续调用该函数,进行处理遍历
      if(  !vertexesMap.get(vertexes[i])  ){
        this.dfsNode(vertexes[i],vertexesMap);
      }
      //4.2 如果处理过则不用管了,
    }

  }
}
let graph = new Graph();
graph.insertVertexes('a');
graph.insertVertexes('b');
graph.insertVertexes('c');
graph.insertVertexes('d');
graph.insertEdge("a",'b');
graph.insertEdge("a",'c');
graph.insertEdge("a",'d');
graph.insertEdge("b",'c');
graph.insertEdge("b",'d');
// graph.bfs();
graph.dfs();

我的个人博客地址:点次进入我的博客网

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 邻接表和邻接矩阵都可以用来实现的深度和广度优先搜索算法以及dijkstra算法。 深度优先搜索算法(DFS)是一种递归的算法,它从的某个顶点开始遍历,尽可能深地搜索,直到找到目标节点或者到达的最深处。邻接表和邻接矩阵都可以用来实现DFS算法。 广度优先搜索算法(BFS)是一种非递归的算法,它从的某个顶点开始遍历,先访问该顶点的所有邻居节点,然后再访问邻居节点的邻居节点,以此类推,直到找到目标节点或者遍历完整个邻接表和邻接矩阵都可以用来实现BFS算法。 Dijkstra算法是一种用于求解最短路径的算法,它可以用邻接矩阵或者邻接表实现。该算法从起点开始,逐步扩展到其他节点,每次选择当前距离起点最近的节点,并更新与该节点相邻的节点的距离。最终得到起点到其他节点的最短路径。 ### 回答2: 是离散数学中一个非常重要的数据结构,其广泛应用于计算机科学、网络科学、金融、生物医学等各个领域。对于的遍历和最短路径查找是论中非常基础和重要的算法。其中,深度优先遍历(DFS)、广度优先遍历(BFS)和迪杰斯特拉算法(Dijkstra)是三种经典的算法,下面对使用邻接表或邻接矩阵实现这三种算法进行详细说明。 1. 邻接表的DFS和BFS算法: 邻接表是一种基于链表表示方法,将每个顶点与其相邻的顶点列表连成一个链表,即每个顶点对应着一个链表,在链表中存储与该顶点相连的所有边。邻接表的DFS和BFS算法可以简单理解为对所有顶点进行遍历,遍历过程中使用栈或队列来记录遍历路径。 具体而言,DFS算法通过递归方式遍历每个节点,从起始节点开始,沿着一条路径走到底,当走到某个节点时,若该节点的相邻节点有未访问的节点,则递归访问该节点的一个未访问的相邻节点,直到所有的节点都被访问为止。BFS算法则通过队列将之前已经访问过的节点从队列中移除,并将该节点的未访问的相邻节点添加到队列末尾,直到队列为空为止。 2. 邻接矩阵的DFS和BFS算法: 邻接矩阵是一种基于二维数组的表示方法,将行和列分别表示中的顶点和边。邻接矩阵的DFS和BFS算法可以看做对邻接矩阵进行深度或广度优先遍历的过程。 具体而言,邻接矩阵的DFS和BFS算法的实现过程与邻接表类似,只不过用邻接矩阵来存储的信息。DFS算法的实现过程中,需要使用栈来记录遍历路径,BFS算法的实现过程中,需要使用队列来记录遍历路径。 3. 邻接表的Dijkstra算法: 邻接表的Dijkstra算法是一种基于贪心策略的最短路径算法。其实现过程主要分为以下几个步骤: (1)通过邻接表来表示带权有向邻接表中每个节点存储该节点的邻接点列表和每个邻接点到该节点的边权值。 (2)设置源点s到每个节点的最短距离数组dis[],初值为无穷大,源点s到 s 的距离为0。 (3)设置源点s到每个节点的前驱路径数组path[],初值为NULL。 (4)将源点s加入集合s{ },并更新dis数组和path数组。 (5)对于其它节点,重复执行以下步骤: a) 在集合V-s{ }中找到距离源点s最近的节点u,并将其加入到s{ }中。 b) 以节点u为中转点,更新节点u的邻接点v到源点s的最短距离dis[v]和前驱路径path[v]。 c) 如果集合V-s{ }为空,算法结束。 4. 邻接矩阵的Dijkstra算法: 邻接矩阵的Dijkstra算法也是一种基于贪心策略的最短路径算法。其实现过程与邻接表类似,只是用邻接矩阵来存储的信息。邻接矩阵的Dijkstra算法的实现可以采用优先队列来提高算法效率。 总而言之,邻接表和邻接矩阵都是常用的表示方法,在实现DFS、BFS和Dijkstra算法时,可以根据具体问题需要选择合适的方法。DFS和BFS算法是遍历的基础,可以通过邻接表和邻接矩阵实现;Dijkstra算法是求解最短路径问题的经典算法,其实现需要采用邻接表或邻接矩阵,通过优先队列可以进一步提高算法效率。 ### 回答3: 邻接表和邻接矩阵是的两种常见的数据结构表示方法。在实现深度和广度优先搜索算法时,可以根据实际情况选择适合的数据结构。如果中有大量的稀疏节点或边,则邻接表更加适合;如果中边和节点数量相对较多,则使用邻接矩阵较为合适。 深度优先搜索算法是一种递归算法,从起始节点开始,不断向下搜索,直到没有相邻未被访问的节点。在实现深度优先搜索算法时,需要建立一个visited数组保存节点是否被访问过的信息,在遍历每个节点时将其visited数组标记为已访问,递归搜索其所有相邻未访问节点,直到所有节点都已被访问。 广度优先搜索算法是一种迭代算法,从起始节点开始,依次遍历每一层节点,直到找到目标节点为止。在实现广度优先搜索算法时,需要建立一个队列来保存已访问的节点,每次取出队列头部节点,遍历其所有相邻未访问节点,并将其加入队列尾部继续搜索。在遍历时,需要保存每个节点的父节点以便最后还原路径。 Dijkstra算法是一种最短路径算法,可以用于有向或无向。在实现Dijkstra算法时,需要建立一个距离数组保存起点到每个节点的距离,另一个visited数组保存节点是否已被访问过。每次从未访问过的节点中选择距离起点最近的节点作为当前节点,遍历其相邻节点,并更新其距离数组。重复这个过程,直到所有节点都被访问过。 以上算法都可以使用邻接表或邻接矩阵进行实现邻接表可以通过数组和链表结合的方式进行实现,每个节点对应一个链表保存其相邻节点。邻接矩阵则可以通过二维数组来实现,其中矩阵中的元素表示两个节点之间是否有边。在具体使用时,需要根据具体情况选择不同的数据结构,以便更好地优化运行效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值