javascript实现二叉搜索树
在使用javascript实现基本的数据结构中,练习了好几周,对基本的数据结构如 栈、队列、链表、集合、哈希表、树、图等内容进行了总结并且写了笔记和代码。
在 github中可以看到 点击查看,可以关注一下我哈。
树的基本术语
二叉树节点的存储结构
创建一个二叉搜索树
二叉树的先序、中序、后续遍历算法
二叉树的非递归先序、中序、后续遍历算法。
文章对树了解的不多的人有点不友好,这里简单介绍(从书上抄下来)那些基本的一点概念吧。
看下面这个示意图
树的基本术语:
结点:A、B、C等都是结点,结点不仅包含数据元素,而且包含指向子树的分支。例如,A结点不仅包含数据元素A、还包含3个指向子树的指针。
结点的度:结点拥有的子树个数或者分支的个数,例如A结点有3棵子树,所以A结点的度为3.
树的度:树中各结点度的最大值。如例子中结点度最大为3(A、D结点)。最小为0(F、G、I、J、K、L、M),所以树的度为3。
叶子节点:又叫做终端节点,指度为0的节点,F、G、I、J、K、L、M节点都是叶子节点。
孩子:结点的子树的根,如A节点的孩子为B、C、D。
双亲:与孩子的定义对应,如B C D结点的双亲都是A。
兄弟:同一个双亲的孩子之间互为兄弟。如B、C、D互为兄弟,因为它们都是A节点的孩子。
祖先:从根到某节点的路径上的所有结点,都是这个节点的祖先。如K的祖先是A B E,因为从A到K的路径为 A---B---E--K。
子孙: 以某节点为根的子树中的所有结点,都是该结点的子孙。如D的子孙为H I J M。
层次:从根开始,根为第一层,根的孩子为第二层,根的孩子的孩子为第三层,以此类推。
树的高度(或者深度):树中结点的最大层次,如例子中的树共有4层,所以高度为4.
理解了上面的树一些基本一些的概念后,我们来看一下什么是二叉树。
1)每个结点最多只有两棵子树,即二叉树中的节点的度只能为0、1、2
2) 子树有左右之分,不能颠倒。
以下二叉树的5中基本状态
构造一个二叉树的节点存储结构
我们会发现,二叉树中的存储结构一个是值,一个是左边有一个,右边有一个。他们分别叫左孩子/左子树 右孩子/右子树。
所以我们会很容易的写出来一个节点的构造函数。
// 树的结构
class BTNode {
constructor() {
this.key = key;
this.lchild = null;
this.rchild = null;
}
}
实现一个二叉搜索树 / 二叉排序树
看一下定义
二叉排序树或者是空树,或者是满足以下性质的二叉树。
1) 若它的左子树不空,则左子树上的所有关键字的值均小于根关键字的值。
2)若它的右子树不空,则右子树上所有关键字的值均大于根关键字的值。
3)左右子树又是一棵二叉排序树。
举个例子
假如要插入一堆数字,数字为 20 10 5 15 13 18 17 30
那么用代码如何实现呢?
let BST = (function () { let ROOT = Symbol(); // 节点结构
let BTNode = class {
constructor(key) {
this.key = key;
this.lchild = null;
this.rchild = null;
}
} // 递归插入节点
let recursionInsert = function (root, node) {
if (node.key < root.key) {
if (root.lchild) {
recursionInsert(root.lchild, node);
} else {
root.lchild = node;
}
} else {
if (root.rchild) {
recursionInsert(root.rchild, node);
} else {
root.rchild = node;
}
}
} // 二叉搜索树类
return class {
constructor() {
this[ROOT] = null;
} // 插入
insert(key) {
let node = new BTNode(key);
let root = this[ROOT];
if (!root) {
this[ROOT] = node;
return;
}
// 递归插入
recursionInsert(root, node);
}
} })(); let bst = new BST(); bst.insert(20);
bst.insert(10);
bst.insert(5);
bst.insert(15);
bst.insert(13);
bst.insert(18);
bst.insert(17);
bst.insert(30); console.log(bst);
在浏览器中一层一层的展开看看是否和我们的一样。
二叉树的遍历算法
二叉树的遍历算法,二叉树的遍历就是按照某种规则将二叉树中的所有数据都访问一遍。
二叉树的遍历方式有先序遍历、中序遍历、后续遍历和层次遍历,很多算法都是基于这几种遍历方式衍生出来的。
递归方式的具体的代码实现
let BST = (function () { let ROOT = Symbol(); // 节点结构
let BTNode = class {
constructor(key) {
this.key = key;
this.lchild = null;
this.rchild = null;
}
} // 递归插入节点
let recursionInsert = function (root, node) {
if (node.key < root.key) {
if (root.lchild) {
recursionInsert(root.lchild, node);
} else {
root.lchild = node;
}
} else {
if (root.rchild) {
recursionInsert(root.rchild, node);
} else {
root.rchild = node;
}
}
}; // 用于中序遍历二叉树的方法
let inorderTraversal = function (root, arr) {
if (!root) return;
inorderTraversal(root.lchild, arr);
arr.push(root.key);
inorderTraversal(root.rchild, arr);
}; // 用于先序遍历的递归函数
let preOrderTraversal = function (root, arr) {
if (!root) return;
arr.push(root.key);
preOrderTraversal(root.lchild, arr);
preOrderTraversal(root.rchild, arr);
}; // 用于后续遍历的递归函数
let lastOrderTraversal = function (root, arr) {
if (!root) return;
lastOrderTraversal(root.lchild, arr);
lastOrderTraversal(root.rchild, arr);
arr.push(root.key);
}; // 二叉搜索树类
return class {
constructor() {
this[ROOT] = null;
} // 插入
insert(key) {
let node = new BTNode(key);
let root = this[ROOT];
if (!root) {
this[ROOT] = node;
return;
}
// 递归插入
recursionInsert(root, node);
} // 中序遍历二叉树
inorderTraversal() {
let arr = [];
inorderTraversal(this[ROOT], arr);
return arr;
} // 先序遍历二叉树
preOrderTraversal() {
let arr = [];
preOrderTraversal(this[ROOT], arr);
return arr;
} // 后续遍历
lastOrderTraversal() {
let arr = [];
lastOrderTraversal(this[ROOT], arr);
return arr;
}
} })(); let bst = new BST(); bst.insert(20);
bst.insert(15);
bst.insert(7);
bst.insert(40);
bst.insert(30);
bst.insert(45);
bst.insert(50); console.log(bst); let a = bst.inorderTraversal();
let b = bst.preOrderTraversal();
let c = bst.lastOrderTraversal(); console.log(a);
console.log(b);
console.log(c);
广度优先遍历
// 广度优先遍历
breadthRirstSearch() {
// 初始化用于广度优先遍历的队列
let queue = new Queue();
console.log('根节点', this[ROOT]); let arr = [];
let root = this[ROOT];
if (!root) return; queue.enqueue(root); while (queue.size()) {
let queueFirst = queue.dequeue();
arr.push(queueFirst.key);
queueFirst.lchild && queue.enqueue(queueFirst.lchild);
queueFirst.rchild && queue.enqueue(queueFirst.rchild);
} return arr;
}
二叉树遍历算法的改进
上面介绍的二傻叔的深度优先遍历算法都是用递归函数实现的,这是很低效的,原因就在于系统帮你调用了一个栈并做了诸如保护现场和恢复现场等复杂的操作,才使得遍历可以用非常简洁的代码实现。
有的人可能会想到,关于二叉树深度优先遍历算法的非递归实现和递归实现,一个是用户自己定义栈,一个是系统栈,为什么用户自己定义的栈要比系统栈执行高效?
一个较为通俗的解释是:递归函数所申请的系统栈,是一个所有递归函数都通用的栈,对于二叉树深度优先遍历算法,系统栈除了记录访问过的节点信息之外,还有其他信息需要记录,以实现函数的递归调用,用户自己定义的栈仅保存了遍历所需的节点信息,是对遍历算法的一个针对性的设计,对于遍历算法来说,显然要比递归函数通用的系统栈更高,也就是一般情况下,专业的要比通用的要好一些。
然而在实际应用中不是这样的,如尾递归在很多机器上都会被优化为循环,因此递归函数不一定就比非递归函数执行效率低。
栈数据结构,满足后进先出的规则用来辅佐我们遍历
// 栈结构 用来辅助非递归遍历
class Stack {
constructor() {
this.items = [];
}
push(data) {
this.items.push(data);
}
pop() {
return this.items.pop();
}
peek() {
return this.items[this.items.length - 1];
}
size() {
return this.items.length;
}
}
非递归的先序遍历
preOrderTraversal() {
console.log('先序遍历');
let root = this[ROOT];
// 初始化辅助遍历存储的栈
let stack = new Stack(); let arr = []; // 用于存储先序遍历的顺序 stack.push(root); // 如果栈不为空 则一直走
while (stack.size()) {
let stackTop = stack.pop(); // 访问栈顶元素
arr.push(stackTop.key);
stackTop.rchild && stack.push(stackTop.rchild);
stackTop.lchild && stack.push(stackTop.lchild);
} return arr;
}
非递归的中序排序
// 中序遍历二叉树
inorderTraversal() {
// 初始化用于辅助排序的栈
let stack = new Stack; let p = null; // 用于指向当前遍历到的元素
let arr = []; // 用户记录排序的顺序
p = this[ROOT]; while (stack.size() || p !== null) {
while (p !== null) {
stack.push(p);
p = p.lchild;
} // 如果栈不为空 出栈
if (stack.size()) {
p = stack.pop();
arr.push(p.key);
p = p.rchild;
}
}
return arr;
}
非递归的后序排序
// 中序遍历二叉树
inorderTraversal() {
// 初始化用于辅助排序的栈
let stack = new Stack; let p = null; // 用于指向当前遍历到的元素
let arr = []; // 用户记录排序的顺序
p = this[ROOT]; while (stack.size() || p !== null) {
while (p !== null) {
stack.push(p);
p = p.lchild;
} // 如果栈不为空 出栈
if (stack.size()) {
p = stack.pop();
arr.push(p.key);
p = p.rchild;
}
}
return arr;
}
想一下,如果我们的插入顺序第一个数非常大,然后后面的数字都是越来越小的会有什么问题产生呢?
下一篇文章讲述这种问题的一个解决方案,平衡二叉树。
javascript实现二叉搜索树的更多相关文章
- 《剑指offer》— JavaScript(26)二叉搜索树与双向链表
二叉搜索树与双向链表 题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. 思路 递归思想:把大问题转换为若干小问题: 由于Ja ...
- 【JavaScript】Leetcode每日一题-二叉搜索树的范围和
[JavaScript]Leetcode每日一题-二叉搜索树的范围和 [题目描述] 给定二叉搜索树的根结点 root,返回值位于范围 [low, high] 之间的所有结点的值的和. 示例1: 输入: ...
- javascript数据结构——写一个二叉搜索树
二叉搜索树就是左侧子节点值比根节点值小,右侧子节点值比根节点值大的二叉树. 照着书敲了一遍. function BinarySearchTree(){ var Node = function(key) ...
- 《剑指offer》— JavaScript(23)二叉搜索树的后序遍历序列
二叉搜索树的后序遍历序列 题目描述 输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果.如果是则输出Yes,否则输出No.假设输入的数组的任意两个数字都互不相同. 相关知识 二叉查找树(B ...
- 数据结构-二叉搜索树的js实现
一.树的相关概念 1.基本概念 子树 一个子树由一个节点和它的后代构成. 节点的度 节点所拥有的子树的个数. 树的度 树中各节点度的最大值 节点的深度 节点的深度等于祖先节点的数量 树的高度 树的高度 ...
- js 二叉搜索树
二叉搜索树:顾名思义,树上每个节点最多只有二根分叉:而且左分叉节点的值 < 右分叉节点的值 . 特点:插入节点.找最大/最小节点.节点值排序 非常方便 1 2 3 4 5 6 7 8 9 10 ...
- 数据结构☞二叉搜索树BST
二叉查找树(Binary Search Tree),(又:二叉搜索树,二叉排序树)它可以是一棵空树,也可以是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值: 若它 ...
- [数据结构]——二叉树(Binary Tree)、二叉搜索树(Binary Search Tree)及其衍生算法
二叉树(Binary Tree)是最简单的树形数据结构,然而却十分精妙.其衍生出各种算法,以致于占据了数据结构的半壁江山.STL中大名顶顶的关联容器--集合(set).映射(map)便是使用二叉树实现 ...
- [LeetCode] Serialize and Deserialize BST 二叉搜索树的序列化和去序列化
Serialization is the process of converting a data structure or object into a sequence of bits so tha ...
随机推荐
- TCP/UDP对比总结
目录 1 TCP-UDP对比 2 UDP介绍 3 TCP介绍 3.1 可靠传输的原理和实现 3.1.1 可靠传输原理 3.1.2 可靠传输实现 3.2 TCP面向连接管理 3.2.1 建立连接 3.2 ...
- 【KakaJSON手册】04_JSON转Model_04_值过滤
在KakaJSON手册的第2篇文章中提过:由于JSON格式能表达的数据类型是比较有限的,所以服务器返回的JSON数据有时无法自动转换成客户端想要的数据类型 比如客户端想要的是Date类型,服务器返回的 ...
- 解决!!-- krb5-libs.x86_64被卸载,yum不能使用,ssh不能连接
常在河边走哪有不湿鞋,常玩服务器哪有不搞挂几台,一不小心就搞挂了 今天删除 krb5-libs.x86_64下了狠功夫..... 用了命令: rpm -e --nodeps krb5-libs.x8 ...
- 苹果电脑基本设置+Linux 命令+Android 实战集锦
本文微信公众号「AndroidTraveler」首发. 背景 大多数应届毕业生在大学期间使用的比较多的是 windows 电脑,因此初入职场如果拿到一台苹果电脑,可能一时间不能够很快的上手.基于此,这 ...
- java120经典面试题
经典面试题 -----version 1.0 题注:以下答案仅限本人个人见解,若有错误和建议请多多指教.QQ:1807812486 题目来源 1.什么是Java虚拟机?为什么Java被称作是" ...
- imwrite imshow机制
今天在做数据增强的时候,遇到一个奇怪的问题.调用imwite生成的图片,在本地用图片查看器打开的时候是正常的.但是在代码里imshow的时候是一片亮白. 代码类似如下 gaussian_img = a ...
- Nginx总结(一)Linux如何安装Nginx
以前写过一些Nginx的文章,但都是用到什么说什么,没有一个完整系统的总结.趁最近有时间,打算将Nginx相关的内容重新整理一下.nginx系列文章地址如下:https://www.cnblogs.c ...
- HBase 系列(五)——HBase 常用 Shell 命令
一.基本命令 打开 Hbase Shell: # hbase shell 1.1 获取帮助 # 获取帮助 help # 获取命令的详细信息 help 'status' 1.2 查看服务器状态 stat ...
- mybatis 源码分析(五)Interceptor 详解
本篇博客将主要讲解 mybatis 插件的主要流程,其中主要包括动态代理和责任链的使用: 一.mybatis 拦截器主体结构 在编写 mybatis 插件的时候,首先要实现 Interceptor 接 ...
- unity之Layer作用
1.使用layer做分层渲染 创建两个物体 添加AB两个层级 分别为两个物体设置对应的层级 为摄像机选择渲染层次(在这个例子中,取消对B层的渲染) 在游戏界面中,将不会显示B层的游戏对象