:非顺序数据结构,对于存储需要快速查找的数据非常有用。
二叉树:二叉树中的节点最多只能有两个子节点(左侧子节点和右侧子节点)。这些定义有助于我们写出更高效的向/从树中插入、查找和删除节点的算法。
二叉搜索树:二叉树的一种,但是它只允许你在左侧节点存储(比父节点)小的值,在右侧节点存储(比父节点)大/等于的值。
 
遍历一棵树:是指访问树的每个节点并对它们进行某种操作的过程。访问树的所有节点有三种方式:中序、先序和后序。
中序遍历:是一种以上行顺序访问 BST 所有节点的的遍历方式,也就是以从最小到最大的顺序访问所有节点。中序遍历的一种应用就是对树进行排序操作。
先序遍历:是以优先于后代节点的顺序访问每个节点的。先序遍历的一种应用是打印一个结构化的文档。
后序遍历:是先访问节点的后代节点,再访问节点本身。后序遍历的一种应用是计算一个目录和它的子目录中所有文件所占空间的大小。
搜索树中的值:在树中,有三种经常执行的搜索类型:最小值、最大值、搜索特定的值
 

BST 存在一个问题:取决于添加的节点树,树的一条边可能会非常深;也就是说,树的一条分支会有很多层,二其他的分支却只有几层。这会在需要在某条边上添加、移除和搜索某个节点时引起一些性能问题。

AVL树:为了解决 BST 的问题,有一种树叫作阿德尔森-维尔斯和兰迪斯树(AVL树)。AVL树是一种自平衡二叉搜索树,意思是任何一个节点两侧子树的高度之差最多为1。也就是说这种树会在添加或移除节点时尽量试着成为一棵完全树。
红黑树:特殊的二叉树,可以进行高效的中序遍历。
 
术语:
         术语             
                                  解释                                 
节点
树中的每个元素
根节点
树顶部的节点,没有父节点
内部节点
至少有一个子节点的节点
外部节点(叶节点)
没有子元素的节点
子树
由节点和它的后代构成
深度
节点的一个属性,取决于它的祖先节点的数量
高度
取决于所有节点深度的最大值

---------------------------------------------------------------------------------------------------------
二叉搜索树方法声明:
1. 和链表一样,通过指针来表示节点之间的关系(术语称其为边);
2. 声明 Node 来表示树中的每个节点。
注意:不同于之前将节点本身称作节点或项,我们会称其为键。键是树相关的术语中对节点的称呼。
序号
方法
说明
1
insert(key) 向树中插入一个新的键
2
search(key) 在树中查找一个键,如果节点存在,则返回 true;如果不存在,则返回 false
3
inOrderTraverse 通过中序遍历方式遍历所有节点
4
preOrderTraverse 通过先序遍历方式遍历所有节点
5
postOrderTraverse 通过后序遍历方式遍历所有节点
6
min
返回树种最小的值/键
7
max 返回树种最大的值/键
8
remove(key) 从树中移除某个键


二叉搜索树的实现:
 function BinarySearchTree() {
// 辅助类Node定义
var Node = function(key) {
this.key = key;
this.left = null;
this.right = null;
}
var root = null; // 向树中插入一个新的键
this.insert = function(key) {
var newNode = new Node(key); if (root === null) {
root = newNode;
} else {
insertNode(root, newNode); // 私有的辅助函数
}
} this.insertNode = function(node, newNode) {
if (newNode.key < node.key) {
if (node.left === null) {
node.left = newNode;
} else {
insertNode(node.left, newNode); // 如果有左侧子节点,则通过递归调用继续找到树的下一层
}
} else {
if (node.right === null) {
node.right = newNode;
} else {
insertNode(node.right, newNode); // 如果有右侧子节点,则通过递归调用继续找到树的下一层
}
}
} /* 中序遍历
* inOrderTraverse方法接收一个回调函数作为参数。
* 回调函数用来定义我们对遍历到的每个节点进行的操作(也叫访问者模式)
*/
this.inOrderTraverse = function(callback) {
inOrderTraverseNode(root, callback);
} this.inOrderTraverseNode = function(node, callback) {
if (node !== null) {
inOrderTraverseNode(node.left, callback);
callback(node.key);
inOrderTraverseNode(node.right, callback);
}
} // 先序遍历
this.preOrderTraverse = function(callback) {
preOrderTraverseNode(root, callback);
} this.preOrderTraverseNode = function(node, callback) {
if (node !== null) {
callback(node.key);
preOrderTraverseNode(node.left, callback);
preOrderTraverseNode(node.right, callback);
}
} // 后序遍历
this.postOrderTraverse = function(allback) {
postOrderTraverseNode(root, callback);
} this.postOrderTraverseNode = function(node, callback) {
if (node !== null) {
postOrderTraverseNode(node.left, callback);
postOrderTraverseNode(node.right, callback);
callback(node.key);
}
} // 搜索树中的最小值
this.min = function() {
return minNode(root);
} var minNode = function(node) {
if (node) {
while (node && node.left !== null) {
node = node.lfet;
}
return node.key;
}
return null;
} // 搜索树中的最大值
this.max = function() {
return maxNode(root);
} this.maxNode = function(node) {
if (node) {
while (node && node.right !== null) {
node = node.right;
}
return node.key;
}
return null;
} // 搜索树中的特定值
this.search = function(key) {
return searchNode(root, key);
} var searchNode = function(node, key) {
if (node === null) {
return false;
}
if (key < node.key) {
return searchNode(node.left ,key);
} else if (key > node.key) {
return searchNode(node.right, key);
} else {
return true;
}
} // 从树中移除某个键
this.remove = function(key) {
root = removeNode(root, key); // root被赋值为removeNode方法的返回值
} var removeNode = function(node, key) {
if (node === null) { // 键不存在于树中,则返回null
return null;
}
if (key < node.key) { // 如果要找的键比当前节点的值小,就沿着树的左边找到下一个节点
node.left = removeNode(node.left, key);
return node;
} else if (key > node.key) { // 如果要找的键比当前节点的值大,就沿着树的左边找到下一个节点
node.right = removeNode(node.right, key);
return node;
} else { // 键等于node.key // 第一种情况——一个叶节点
if (node.left === null && node.right === null) {
node = null;
return node;
} // 第二种情况——一个只有一个子节点的节点
if (node.left === null) {
node = node.right;
return node;
} else if (node.right === null) {
node = node.left;
return node;
} // 第三种情况——一个有两个子节点的节点
var aux = findMinNode(node.right);
node.key = aux.key; // 用右侧子树中的最小节点的键去更新这个节点的值
node.right = removeNode(node.right, aux.key); // 移除右侧子树中的最小节点
return node;
}
}
}

BinarySearchTree.js

 
 参考书籍:《学习JavaScript数据结构与算法》 
 

JavaScript数据结构——树的更多相关文章

  1. JavaScript数据结构——树的实现

    在计算机科学中,树是一种十分重要的数据结构.树被描述为一种分层数据抽象模型,常用来描述数据间的层级关系和组织结构.树也是一种非顺序的数据结构.下图展示了树的定义: 在介绍如何用JavaScript实现 ...

  2. JavaScript数据结构-树

    我认为这社会上,也不差钱好多人,可能好多人也不差权力.可是我认为能得到这样的满足的也不多. –郭小平<临汾红丝带学校校长> ​ 树是计算机科学中经经常使用到的一种数据结构. 树是一种非线性 ...

  3. 学习javascript数据结构(四)——树

    前言 总括: 本文讲解了数据结构中的[树]的概念,尽可能通俗易懂的解释树这种数据结构的概念,使用javascript实现了树,如有纰漏,欢迎批评指正. 原文博客地址:学习javascript数据结构( ...

  4. 为什么我要放弃javaScript数据结构与算法(第八章)—— 树

    之前介绍了一些顺序数据结构,介绍的第一个非顺序数据结构是散列表.本章才会学习另一种非顺序数据结构--树,它对于存储需要快速寻找的数据非常有用. 本章内容 树的相关术语 创建树数据结构 树的遍历 添加和 ...

  5. JavaScript 数据结构与算法之美 - 非线性表中的树、堆是干嘛用的 ?其数据结构是怎样的 ?

    1. 前言 想学好前端,先练好内功,内功不行,就算招式练的再花哨,终究成不了高手. 非线性表(树.堆),可以说是前端程序员的内功,要知其然,知其所以然. 笔者写的 JavaScript 数据结构与算法 ...

  6. javascript数据结构与算法-- 二叉树

    javascript数据结构与算法-- 二叉树 树是计算机科学中经常用到的一种数据结构.树是一种非线性的数据结构,以分成的方式存储数据,树被用来存储具有层级关系的数据,比如文件系统的文件,树还被用来存 ...

  7. 为什么我要放弃javaScript数据结构与算法(第九章)—— 图

    本章中,将学习另外一种非线性数据结构--图.这是学习的最后一种数据结构,后面将学习排序和搜索算法. 第九章 图 图的相关术语 图是网络结构的抽象模型.图是一组由边连接的节点(或顶点).学习图是重要的, ...

  8. 为什么我要放弃javaScript数据结构与算法(第七章)—— 字典和散列表

    本章学习使用字典和散列表来存储唯一值(不重复的值)的数据结构. 集合.字典和散列表可以存储不重复的值.在集合中,我们感兴趣的是每个值本身,并把它作为主要元素.而字典和散列表中都是用 [键,值]的形式来 ...

  9. JavaScript数据结构——字典和散列表的实现

    在前一篇文章中,我们介绍了如何在JavaScript中实现集合.字典和集合的主要区别就在于,集合中数据是以[值,值]的形式保存的,我们只关心值本身:而在字典和散列表中数据是以[键,值]的形式保存的,键 ...

随机推荐

  1. mysql sql 百万级数据库优化方案

    1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索 ...

  2. Redis persistence demystified - part 1

    关于Redis我的一部分工作是阅读博客,论坛以及twitter时间线(time line).对于开发者来说,能够了解用户社区,非用户社区如果理解他正在开发的产品是非常重要的.据我所知,持久化特性是最易 ...

  3. cocopods的使用方法

    虽然网上关于CocoaPods安装教程多不胜数,但是我在安装的过程中还是出现了很多错误,所以大家可以照下来步骤装一下,我相信会很好用. 前言 在iOS项目中使用第三方类库可以说是非常常见的事,但是要正 ...

  4. HDU 1394 Minimum Inversion Number

    //============================================================================ // Name : B.cpp // Au ...

  5. C#同一位置切换显示两个Panel内容

    如果两个panel重合在一起,点击不同按钮切换显示不同的panel,需要xxx.BringToFront(); 1.首先让两个panel的visible都为false, 在加载页面load方法里可以让 ...

  6. (BFS)hdoj2377-Bus Pass

    题目地址 因为最后要看的是到所有路线上的区域最大距离最小的中心点,所以可以采取遍历路线上所有的区域,对每个区域进行BFS的办法.为了更方便的在每一次BFS都遍历所有的区域,可以加一个reach数组,记 ...

  7. C语言:文件操作

    以附加方式打开文件,输入数据,关闭文件. #include<stdio.h> #include<stdlib.h> int main() { FILE *fp = NULL; ...

  8. UINavigationController 子控制器管理原理

    UINavigationController 显示在导航控制器上的控制器 永远是栈顶控制器 后进先出  先进后出原则 /** * 程序获得焦点才能获取触摸事件 * */- (void)applicat ...

  9. T420修改wifi灯闪动模式

    给T420新装了centos7发现默认的配置wifi灯是工作时闪动的,有点晃眼,想改成简单的on 的时候常亮,off的时候常暗的模式 添加配置文件: vi /etc/modprobe.d/wlanle ...

  10. HTML特殊符号对应代码

    HTML中的特殊符号所对应代码,收藏使用 符号 HTML 符号 HTML     & & < < > > ⁄ ⁄ " " ¸ ¸ ° ° ½ ...