面试官:说说你对树的理解?相关的操作有哪些?

开发 架构
在计算机领域,树形数据结构是一类重要的非线性数据结构,可以表示数据之间一对多的关系。以树与二叉树最为常用,直观看来,树是以分支关系定义的层次结构。

[[425982]]

本文转载自微信公众号「JS每日一题」,作者灰灰。转载本文请联系JS每日一题公众号。

一、是什么

在计算机领域,树形数据结构是一类重要的非线性数据结构,可以表示数据之间一对多的关系。以树与二叉树最为常用,直观看来,树是以分支关系定义的层次结构

二叉树满足以下两个条件:

  • 本身是有序树
  • 树中包含的各个节点的度不能超过 2,即只能是 0、1 或者 2

如下图,左侧的为二叉树,而右侧的因为头结点的子结点超过2,因此不属于二叉树:

同时,二叉树可以继续进行分类,分成了满二叉树和完成二叉树:

满二叉树:如果二叉树中除了叶子结点,每个结点的度都为 2

完成二叉树:如果二叉树中除去最后一层节点为满二叉树,且最后一层的结点依次从左到右分布

二、操作

关于二叉树的遍历,常见的有:

  • 前序遍历
  • 中序遍历
  • 后序遍历
  • 层序遍历

前序遍历

前序遍历的实现思想是:

  • 访问根节点
  • 访问当前节点的左子树
  • 若当前节点无左子树,则访问当前节点的右子

根据遍历特性,递归版本用代码表示则如下:

  1. const preOrder = (root) => { 
  2.   if(!root){ return } 
  3.   console.log(root) 
  4.   preOrder(root.left
  5.   preOrder(root.right

如果不使用递归版本,可以借助栈先进后出的特性实现,先将根节点压入栈,再分别压入右节点和左节点,直到栈中没有元素,如下:

  1. const preOrder = (root) => { 
  2.   if(!root){ return } 
  3.   const stack = [root] 
  4.   while (stack.length) { 
  5.     const n = stack.pop() 
  6.     console.log(n.val) 
  7.     if (n.right) { 
  8.       stack.push(n.right
  9.     } 
  10.     if (n.left) { 
  11.       stack.push(n.left
  12.     } 
  13.   } 

中序遍历

前序遍历的实现思想是:

  • 访问当前节点的左子树
  • 访问根节点
  • 访问当前节点的右子

递归版本很好理解,用代码表示则如下:

  1. const inOrder = (root) => { 
  2.   if (!root) { return } 
  3.   inOrder(root.left
  4.   console.log(root.val) 
  5.   inOrder(root.right

非递归版本也是借助栈先进后出的特性,可以一直首先一直压入节点的左元素,当左节点没有后,才开始进行出栈操作,压入右节点,然后有依次压入左节点,如下:

  1. const inOrder = (root) => { 
  2.   if (!root) { return } 
  3.   const stack = [root] 
  4.   let p = root 
  5.   while(stack.length || p){ 
  6.     while (p) { 
  7.       stack.push(p) 
  8.       p = p.left 
  9.     } 
  10.     const n = stack.pop() 
  11.     console.log(n.val) 
  12.     p = n.right 
  13.   } 

后序遍历

前序遍历的实现思想是:

  • 访问当前节点的左子树
  • 访问当前节点的右子
  • 访问根节点

递归版本,用代码表示则如下:

  1. const postOrder = (root) => { 
  2.   if (!root) { return } 
  3.   postOrder(root.left
  4.   postOrder(root.right
  5.   console.log(n.val) 
  6.  } 

后序遍历非递归版本实际跟全序遍历是逆序关系,可以再多创建一个栈用来进行输出,如下:

  1. const preOrder = (root) => { 
  2.   if(!root){ return } 
  3.   const stack = [root] 
  4.   const outPut = [] 
  5.   while (stack.length) { 
  6.     const n = stack.pop() 
  7.     outPut.push(n.val) 
  8.     if (n.right) { 
  9.       stack.push(n.right
  10.     } 
  11.     if (n.left) { 
  12.       stack.push(n.left
  13.     } 
  14.   } 
  15.   while (outPut.length) { 
  16.     const n = outPut.pop() 
  17.     console.log(n.val) 
  18.   } 

层序遍历

按照二叉树中的层次从左到右依次遍历每层中的结点

借助队列先进先出的特性,从树的根结点开始,依次将其左孩子和右孩子入队。而后每次队列中一个结点出队,都将其左孩子和右孩子入队,直到树中所有结点都出队,出队结点的先后顺序就是层次遍历的最终结果

用代码表示则如下:

  1. const levelOrder = (root) => { 
  2.     if (!root) { return [] } 
  3.     const queue = [[root, 0]] 
  4.     const res = [] 
  5.     while (queue.length) { 
  6.         const n = queue.shift() 
  7.         const [node, leval] = n 
  8.         if (!res[leval]) { 
  9.             res[leval] = [node.val] 
  10.         } else { 
  11.             res[leval].push(node.val) 
  12.         } 
  13.         if (node.left) { queue.push([node.left, leval + 1]) } 
  14.         if (node.right) { queue.push([node.right, leval + 1]) } 
  15.     } 
  16.     return res 
  17. }; 

三、总结

树是一个非常重要的非线性结构,其中二叉树以二叉树最常见,二叉树的遍历方式可以分成前序遍历、中序遍历、后序遍历

同时,二叉树又分成了完成二叉树和满二叉树

参考文献

https://baike.baidu.com/item/%E4%BA%8C%E5%8F%89%E6%A0%91

 

http://data.biancheng.net/view/27.html

 

责任编辑:武晓燕 来源: JS每日一题
相关推荐

2021-09-26 10:57:16

集合操作场景

2021-08-20 08:33:19

操作系统OS

2021-08-09 07:47:40

Git面试版本

2021-11-25 10:18:42

RESTfulJava互联网

2020-12-01 08:47:36

Java异常开发

2020-06-12 15:50:56

options前端服务器

2021-09-09 07:21:26

TypeScript 高级类型

2021-09-16 07:52:18

算法应用场景

2019-05-10 10:50:04

Spring AOPJDK动态代理CGLIB动态代理

2021-11-02 22:04:58

模式

2021-11-03 14:10:28

工厂模式场景

2022-02-21 17:24:18

序列化对象存储

2021-11-10 07:47:49

组合模式场景

2021-08-16 08:33:26

git

2020-12-04 06:27:04

序列化面试官Java

2021-11-09 08:51:13

模式命令面试

2021-11-05 07:47:56

代理模式对象

2024-07-26 08:10:10

2021-09-29 07:24:20

场景数据

2021-10-19 08:07:21

微信小程序代码
点赞
收藏

51CTO技术栈公众号