iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么理解Java优先遍历和广度优先遍历算法
  • 777
分享到

怎么理解Java优先遍历和广度优先遍历算法

2023-06-16 13:06:49 777人浏览 八月长安
摘要

这篇文章主要讲解了“怎么理解Java优先遍历和广度优先遍历算法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解Java优先遍历和广度优先遍历算法”吧!深度优先遍历主要思路是从图中一个未

这篇文章主要讲解了“怎么理解Java优先遍历和广度优先遍历算法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解Java优先遍历和广度优先遍历算法”吧!

深度优先遍历

主要思路是从图中一个未访问的顶点 V  开始,沿着一条路一直走到底,然后从这条路尽头的节点回退到上一个节点,再从另一条路开始走到底...,不断递归重复此过程,直到所有的顶点都遍历完成,它的特点是不撞南墙不回头,先走完一条路,再换一条路继续走。

树是图的一种特例(连通无环的图就是树),接下来我们来看看树用深度优先遍历该怎么遍历。

怎么理解Java优先遍历和广度优先遍历算法

我们从根节点 1 开始遍历,它相邻的节点有 2,3,4,先遍历节点 2,再遍历 2 的子节点 5,然后再遍历 5 的子节点 9。

怎么理解Java优先遍历和广度优先遍历算法

上图中一条路已经走到底了(9是叶子节点,再无可遍历的节点),此时就从 9 回退到上一个节点 5,看下节点 5 是否还有除 9  以外的节点,没有继续回退到 2,2 也没有除 5 以外的节点,回退到 1,1 有除 2 以外的节点 3,所以从节点 3 开始进行深度优先遍历,如下:

怎么理解Java优先遍历和广度优先遍历算法

同理从 10 开始往上回溯到 6, 6 没有除 10 以外的子节点,再往上回溯,发现 3 有除 6 以外的子点 7,所以此时会遍历 7。

怎么理解Java优先遍历和广度优先遍历算法

从 7 往上回溯到 3, 1,发现 1 还有节点 4 未遍历,所以此时沿着 4, 8 进行遍历,这样就遍历完成了。

完整的节点的遍历顺序如下(节点上的的蓝色数字代表):

怎么理解Java优先遍历和广度优先遍历算法

相信大家看到以上的遍历不难发现这就是树的前序遍历,实际上不管是前序遍历,还是中序遍历,亦或是后序遍历,都属于深度优先遍历。

那么深度优先遍历该怎么实现呢,有递归和非递归两种表现形式,接下来我们以二叉树为例来看下如何分别用递归和非递归来实现深度优先遍历。

1、递归实现

递归实现比较简单,由于是前序遍历,所以我们依次遍历当前节点,左节点,右节点即可,对于左右节点来说,依次遍历它们的左右节点即可,依此不断递归下去,直到叶节点(递归终止条件),代码如下:

public class Solution {     private static class node {                  public int value;                  public Node left;                  public Node right;          public Node(int value, Node left, Node right) {             this.value = value;             this.left = left;             this.right = right;         }     }      public static void dfs(Node treeNode) {         if (treeNode == null) {             return;         }         // 遍历节点         process(treeNode)         // 遍历左节点         dfs(treeNode.left);         // 遍历右节点         dfs(treeNode.right);     } }

递归的表达性很好,也很容易理解,不过如果层级过深,很容易导致栈溢出。所以我们重点看下非递归实现。

2、非递归实现

仔细观察深度优先遍历的特点,对二叉树来说,由于是先序遍历(先遍历当前节点,再遍历左节点,再遍历右节点),所以我们有如下思路:

对于每个节点来说,先遍历当前节点,然后把右节点压栈,再压左节点(这样弹栈的时候会先拿到左节点遍历,符合深度优先遍历要求)。

弹栈,拿到栈顶的节点,如果节点不为空,重复步骤 1, 如果为空,结束遍历。

我们以以下二叉树为例来看下如何用栈来实现 DFS。

怎么理解Java优先遍历和广度优先遍历算法

整体动图如下:

怎么理解Java优先遍历和广度优先遍历算法

整体思路还是比较清晰的,使用栈来将要遍历的节点压栈,然后出栈后检查此节点是否还有未遍历的节点,有的话压栈,没有的话不断回溯(出栈),有了思路,不难写出如下用栈实现的二叉树的深度优先遍历代码:

 public static void dfsWithStack(Node root) {     if (root == null) {         return;     }      Stack<Node> stack = new Stack<>();     // 先把根节点压栈     stack.push(root);     while (!stack.isEmpty()) {         Node treeNode = stack.pop();         // 遍历节点         process(treeNode)          // 先压右节点         if (treeNode.right != null) {             stack.push(treeNode.right);         }          // 再压左节点         if (treeNode.left != null) {             stack.push(treeNode.left);         }     } }

可以看到用栈实现深度优先遍历其实代码也不复杂,而且也不用担心递归那样层级过深导致的栈溢出问题。

广度优先遍历

广度优先遍历,指的是从图的一个未遍历的节点出发,先遍历这个节点的相邻节点,再依次遍历每个相邻节点的相邻节点。

上文所述树的广度优先遍历动图如下,每个节点的值即为它们的遍历顺序。所以广度优先遍历也叫层序遍历,先遍历第一层(节点 1),再遍历第二层(节点  2,3,4),第三层(5,6,7,8),第四层(9,10)。

怎么理解Java优先遍历和广度优先遍历算法

深度优先遍历用的是栈,而广度优先遍历要用队列来实现,我们以下图二叉树为例来看看如何用队列来实现广度优先遍历。

怎么理解Java优先遍历和广度优先遍历算法

动图如下:

怎么理解Java优先遍历和广度优先遍历算法

相信看了以上动图,不难写出如下代码:

 private static void bfs(Node root) {     if (root == null) {         return;     }     Queue<Node> stack = new LinkedList<>();     stack.add(root);      while (!stack.isEmpty()) {         Node node = stack.poll();         System.out.println("value = " + node.value);         Node left = node.left;         if (left != null) {             stack.add(left);         }         Node right = node.right;         if (right != null) {             stack.add(right);         }     } }

习题演练

接下来我们来看看在 LeetCode 中出现的一些使用 DFS,BFS 来解题的题目:

leetcode 104,111: 给定一个二叉树,找出其最大/最小深度。

例如:给定二叉树 [3,9,20,null,null,15,7],

   3   / \  9  20    /  \   15   7

则它的最小深度 2,最大深度 3。

解题思路:这题比较简单,只不过是深度优先遍历的一种变形,只要递归求出左右子树的最大/最小深度即可,深度怎么求,每递归调用一次函数,深度加一。不难写出如下代码:

 public static int getMaxDepth(Node node) {     if (node == null) {         return 0;     }     int leftDepth = getMaxDepth(node.left) + 1;     int rightDepth = getMaxDepth(node.right) + 1;     return Math.max(leftDepth, rightDepth); }   public static int getMinDepth(Node node) {     if (node == null) {         return 0;     }     int leftDepth = getMinDepth(node.left) + 1;     int rightDepth = getMinDepth(node.right) + 1;     return Math.min(leftDepth, rightDepth); }

leetcode 102:  给你一个二叉树,请你返回其按层序遍历得到的节点值。(即逐层地,从左到右访问所有节点)。示例,给定二叉树:[3,9,20,null,null,15,7]。

   3   / \  9  20    /  \   15   7

返回其层次遍历结果:

[   [3],   [9,20],   [15,7] ]

解题思路:显然这道题是广度优先遍历的变种,只需要在广度优先遍历的过程中,把每一层的节点都添加到同一个数组中即可,问题的关键在于遍历同一层节点前,必须事先算出同一层的节点个数有多少(即队列已有元素个数),因为  BFS 用的是队列来实现的,遍历过程中会不断把左右子节点入队,这一点切记!动图如下:

怎么理解Java优先遍历和广度优先遍历算法

根据以上动图思路不难得出代码如下:

Java 代码

 private static List<List<Integer>> bfsWithBinaryTreeLevelOrderTraversal(Node root) {     if (root == null) {         // 根节点为空,说明二叉树不存在,直接返回空数组         return Arrays.asList();     }      // 最终的层序遍历结果     List<List<Integer>> result = new ArrayList<>();      Queue<Node> queue = new LinkedList<>();     queue.offer(root);      while (!queue.isEmpty()) {         // 记录每一层         List<Integer> level = new ArrayList<>();         int levelNum = queue.size();         // 遍历当前层的节点         for (int i = 0; i < levelNum; i++) {             Node node = queue.poll();             // 队首节点的左右子节点入队,由于 levelNum 是在入队前算的,所以入队的左右节点并不会在当前层被遍历到             if (node.left != null) {                 queue.add(node.left);             }             if (node.right != null) {                 queue.add(node.right);             }             level.add(node.value);         }         result.add(level);     }      return result; }

Python 代码

class Solution:     def levelOrder(self, root):         """         :type root: TreeNode         :rtype: List[List[int]]         """         res = []  #嵌套列表,保存最终结果         if root is None:             return res                  from collections import deque         que = deque([root])  #队列,保存待处理的节点         while len(que)!=0:             lev = []  #列表,保存该层的节点的值             thislevel = len(que)  #该层节点个数             while thislevel!=0:                 head = que.popleft()  #弹出队首节点                 #队首节点的左右孩子入队                 if head.left is not None:                     que.append(head.left)                 if head.right is not None:                     que.append(head.right)                 lev.append(head.val)  #队首节点的值压入本层                 thislevel-=1             res.append(lev)         return res

这题用 BFS 是显而易见的,但其实也可以用 DFS, 如果在面试中能用 DFS 来处理,会是一个比较大的亮点。

用 DFS 怎么处理呢,我们知道, DFS  可以用递归来实现,其实只要在递归函数上加上一个「层」的变量即可,只要节点属于这一层,则把这个节点放入相当层的数组里,代码如下:

private static final List<List<Integer>> TRAVERSAL_LIST  = new ArrayList<>();  private static void dfs(Node root, int level) {     if (root == null) {         return;     }      if (TRAVERSAL_LIST.size() < level + 1) {         TRAVERSAL_LIST.add(new ArrayList<>());     }      List<Integer> levelList = TRAVERSAL_LIST.get(level);     levelList.add(root.value);      // 遍历左结点     dfs(root.left, level + 1);      // 遍历右结点     dfs(root.right, level + 1); }

DFS,BFS 在搜索引擎中的应用我们几乎每天都在 Google, Baidu 这些搜索引擎,那大家知道这些搜索引擎是怎么工作的吗,简单来说有三步:

1、网页抓取

搜索引擎通过爬虫将网页爬取,获得页面 html 代码存入数据库

2、预处理

索引程序对抓取来的页面数据进行文字提取,中文分词,(倒排)索引等处理,以备排名程序使用

3、排名

用户输入关键词后,排名程序调用索引数据库数据,计算相关性,然后按一定格式生成搜索结果页面。

我们重点看下第一步,网页抓取。

这一步的大致操作如下:给爬虫分配一组起始的网页,我们知道网页里其实也包含了很多超链接,爬虫爬取一个网页后,解析提取出这个网页里的所有超链接,再依次爬取出这些超链接,再提取网页超链接。。。,如此不断重复就能不断根据超链接提取网页。如下图示:

怎么理解Java优先遍历和广度优先遍历算法

如上所示,最终构成了一张图,于是问题就转化为了如何遍历这张图,显然可以用深度优先或广度优先的方式来遍历。

如果是广度优先遍历,先依次爬取第一层的起始网页,再依次爬取每个网页里的超链接,如果是深度优先遍历,先爬取起始网页  1,再爬取此网页里的链接...,爬取完之后,再爬取起始网页 2...

实际上爬虫是深度优先与广度优先两种策略一起用的,比如在起始网页里,有些网页比较重要(权重较高),那就先对这个网页做深度优先遍历,遍历完之后再对其他(权重一样的)起始网页做广度优先遍历。

感谢各位的阅读,以上就是“怎么理解Java优先遍历和广度优先遍历算法”的内容了,经过本文的学习后,相信大家对怎么理解Java优先遍历和广度优先遍历算法这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: 怎么理解Java优先遍历和广度优先遍历算法

本文链接: https://www.lsjlt.com/news/283772.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么理解Java优先遍历和广度优先遍历算法
    这篇文章主要讲解了“怎么理解Java优先遍历和广度优先遍历算法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解Java优先遍历和广度优先遍历算法”吧!深度优先遍历主要思路是从图中一个未...
    99+
    2023-06-16
  • JavaScript中深度优先遍历和广度优先遍历算法的示例分析
    这篇文章主要为大家展示了“JavaScript中深度优先遍历和广度优先遍历算法的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“JavaScript中深度...
    99+
    2024-04-02
  • Java遍历树深度优先和广度优先的方法是什么
    这篇文章主要介绍了Java遍历树深度优先和广度优先的方法是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java遍历树深度优先和广度优先的方法是什么文章都会有所收获,下面我们一起来看看吧。在编程生活中,我们...
    99+
    2023-07-05
  • Java中怎么实现广度优先遍历
    今天小编给大家分享一下Java中怎么实现广度优先遍历的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。什么是广度优先广度就是扩展...
    99+
    2023-06-29
  • C++实现广度优先遍历图
    本文实例为大家分享了C++实现广度优先遍历图的具体代码,供大家参考,具体内容如下 广度优先遍历 void bfs(int start, int parent[], int di...
    99+
    2024-04-02
  • Java实现广度优先遍历的示例详解
    目录什么是广度优先一个简单的例子程序实现总结什么是广度优先 广度就是扩展开,广度优先的意思就是尽量扩展开。所以在算法实现的时候,就是一个循环遍历枚举每一个邻接点。其基本思路就是按层扩...
    99+
    2024-04-02
  • Java深度优先遍历是什么
    Java深度优先遍历是一种图遍历算法,它通过递归地访问图中的节点,从一个节点开始,沿着一条路径尽可能深入地访问,直到达到不能再深入的...
    99+
    2023-08-11
    Java
  • 简单谈谈Java遍历树深度优先和广度优先的操作方式
    目录1、深度优先2、广度优先在编程生活中,我们总会遇见树性结构,这几天刚好需要对树形结构操作,就记录下自己的操作方式以及过程。现在假设有一颗这样树,(是不是二叉树都没关系,原理都是一...
    99+
    2023-03-24
    java遍历树形数据 java如何遍历树 Java遍历树形结构
  • Python怎么使用广度优先搜索遍历混乱地铁
    这篇文章主要介绍“Python怎么使用广度优先搜索遍历混乱地铁”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Python怎么使用广度优先搜索遍历混乱地铁”文章能帮助大家解决问题。混乱地铁问题【问题描...
    99+
    2023-07-05
  • 怎么用Java深度优先遍历解决迷宫问题
    本文小编为大家详细介绍“怎么用Java深度优先遍历解决迷宫问题”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么用Java深度优先遍历解决迷宫问题”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。什么是深度优先什么...
    99+
    2023-06-29
  • python迷宫问题深度优先遍历实例
    一、迷宫介绍 用python解迷宫问题,迷宫是一个二维列表,本次用深度优先解开迷宫问题。定义起点和终点,从一个位置到下一个位置只能通过向上或下或左或右,走一步来实现,从起点出发,如...
    99+
    2024-04-02
  • Python使用广度优先搜索遍历混乱地铁问题
    目录混乱地铁问题【问题描述】【输入形式】【输出形式】【样例输入】【样例输出】程序设计 总结 混乱地铁问题 【问题描述】 在某个城市中地铁网极度混乱。一条地铁线路上...
    99+
    2023-05-14
    Python混乱地铁 广度优先搜索遍历 混乱地铁
  • 详解Java利用深度优先遍历解决迷宫问题
    目录什么是深度优先一个简单的例子程序实现什么是深度优先 什么是深度,即向下,深度优先,即向下优先,一口气走到底,走到底发现没路再往回走。 在算法实现上来讲,深度优先可以考虑是递归的代...
    99+
    2024-04-02
  • 广度优先遍历与最短路径(Java 实例代码源码包下载)
    目录   广度优先遍历与最短路径 Java 实例代码 src/runoob/graph/ShortestPath.java 文件代码:   广度优先遍历与最短路径 广度优先遍历从某个顶点 v 出发,首先访问这个结点,并将其标记为已访问过,...
    99+
    2023-09-01
    python 算法 开发语言
  • java广度优先算法是什么
    Java广度优先算法是一种用于图的遍历的算法。广度优先搜索(BFS)是一种基于队列的搜索算法,用于在图或树数据结构中遍历或搜索。该算...
    99+
    2023-08-11
    java
  • Java实现深度优先搜索(DFS)和广度优先搜索(BFS)算法
    目录一.深度优先遍历和广度优先遍历1.深度优先遍历2.广度优先遍历二.图像渲染1.题目描述2.问题分析3代码实现1.广度优先遍历2.深度优先遍历三.岛屿的最大面积1.题目描述2.问题...
    99+
    2023-05-18
    Java深度优先和广度优先 Java深度优先 Java广度优先
  • Java编程实现深度优先遍历与连通分量代码示例
    深度优先遍历深度优先遍历类似于一个人走迷宫:如图所示,从起点开始选择一条边走到下一个顶点,没到一个顶点便标记此顶点已到达。当来到一个标记过的顶点时回退到上一个顶点,再选择一条没有到达过的顶点。当回退到的路口已没有可走的通道时继续回退。而连通...
    99+
    2023-05-30
    java 深度优先遍历 连通分量
  • Python 算法基础篇:深度优先搜索( DFS )和广度优先搜索( BFS )
    Python 算法基础篇:深度优先搜索( DFS )和广度优先搜索( BFS ) 引言 1. 深度优先搜索( DFS )算法概述2. 深度优先搜索( DFS )算法实现实例1:图的 DFS 遍...
    99+
    2023-10-04
    深度优先 算法 python 广度优先
  • Java分别利用深度优先和广度优先求解迷宫路径
    目录深度优先实现效果示例代码广度优先实现效果示例代码知识点总结深度优先 实现效果 示例代码 import java.awt.*; import javax.swing.*; p...
    99+
    2024-04-02
  • C++怎么实现由先序和中序遍历二叉树
    这篇文章主要介绍了C++怎么实现由先序和中序遍历二叉树的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++怎么实现由先序和中序遍历二叉树文章都会有所收获,下面我们一起来看看吧。由先序和中序遍历建立二叉树Give...
    99+
    2023-06-19
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作