常见例子:如公司的组织架构图
二叉树: 只能有左右两个子节点
二叉搜索树: 二叉树的一种,但是只允许在左侧节点存储(比父节点)小的值,在右侧节点存储(比父节点)大的值(如上面的图)
创建Node类来表示二叉搜索树中的每个节点:
class Node {
constructor(key) {
// 节点值,就是下图的键
this.key = key;
// 左侧子节点
this.left = null;
// 右侧子节点
this.right = null;
}
toString() {
return `${
this.key}`;
}
}
声明BinarySearchTree类的基本结构:
const Compare = {
LESS_THAN: -1,
BIGGER_THAN: 1,
EQUALS: 0
};
function defaultCompare(a, b) {
if (a === b) {
return Compare.EQUALS;
}
return a < b ? Compare.LESS_THAN : Compare.BIGGER_THAN;
}
class BinarySearchTree {
constructor(compareFn = defaultCompare) {
// 用来比较节点值
this.compareFn = compareFn;
// Node类型的根节点
this.root = null;
}
}
常用方法:
insert(key) {
// 验证是否为特殊情况,即要插入的节点是不是第一个节点
if (this.root === null) {
// 创建一个Node类的实例并将它赋值给root属性来讲root指向新节点
this.root = new Node(key);
}
// 将节点添加到根节点以外的位置,需要借助辅助方法
else {
this.insertNode(this.root, key);
}
}
insertNode(node, key) {
// 如果新节点的键小于当前节点的键,需要检查当前节点的左侧子节点
if (this.compareFn(key, node.key) === Compare.LESS_THAN) {
// 如果没有左侧子节点,就插入新节点
if (node.left == null) {
node.left = new Node(key);
}
// 如果有左侧子节点,就递归调用insertNode方法,继续找到树的下一层
else {
this.insertNode(node.left, key);
}
}
// 如果新节点的键大于当前节点的键
else {
// 如果没有右侧子节点,就插入新节点
if (node.right == null) {
node.right = new Node(key);
}
// 如果有右侧子节点,就递归调用insertNode方法,继续找到树的下一层
else {
this.insertNode(node.right, key);
}
}
}
执行下面代码:
const tree = new BinarySearchTree();
tree.insert(11);
// 插入元素
tree.insert(7);
tree.insert(15);
tree.insert(5);
tree.insert(3);
tree.insert(9);
tree.insert(8);
tree.insert(10);
tree.insert(13);
tree.insert(12);
tree.insert(14);
tree.insert(20);
tree.insert(18);
tree.insert(25);
tree.insert(6);
访问树的三种方式:中序、先序和后序。
中序遍历是一种以上行顺序访问BST所有节点的遍历方式,也就是以从最小到最大的顺序访问所有节点。
inOrderTraverse(callback) {
// 接收一个节点和对应的回调函数作为参数
this.inOrderTraverseNode(this.root, callback);
}
inOrderTraverseNode(node, callback) {
// 先检查以参数形式传入的节点是否为null,这是停止递归继续执行的判断条件
if (node != null) {
// 调用相同的函数来访问左侧子节点
this.inOrderTraverseNode(node.left, callback);
// 对根节点进行操作
callback(node.key);
// 再访问右侧子节点
this.inOrderTraverseNode(node.right, callback);
}
}
执行方法:
const printNode = (value) => console.log(value);
tree.inOrderTraverse(printNode);
先序遍历是以优先于后代节点的顺序访问每个节点的。先序遍历的一种应用是打印一个结构化的文档。
preOrderTraverse(callback) {
this.preOrderTraverseNode(this.root, callback);
}
preOrderTraverseNode(node, callback) {
if (node != null) {
// 先访问节点本身
callback(node.key);
// 再访问左侧节点
this.preOrderTraverseNode(node.left, callback);
// 最后是右侧节点
this.preOrderTraverseNode(node.right, callback);
}
}
postOrderTraverse(callback) {
this.postOrderTraverseNode(this.root, callback);
}
postOrderTraverseNode(node, callback) {
if (node != null) {
// 先访问左侧子节点
this.postOrderTraverseNode(node.left, callback);
// 然后访问右侧子节点
this.postOrderTraverseNode(node.right, callback);
// 最后访问根节点
callback(node.key);
}
}
min() {
// 在调用minNode方法的时候传入树的根节点
return this.minNode(this.root);
}
minNode(node) {
let current = node;
// 遍历树的左边,直到找到树的最下层(最左端)
while (current != null && current.left != null) {
current = current.left;
}
return current;
}
max() {
return this.maxNode(this.root);
}
maxNode(node) {
let current = node;
// 沿着树的右边进行遍历,直到找到最右端的节点
while (current != null && current.right != null) {
current = current.right;
}
return current;
}
因此,对于寻找最小值,总是沿着树的左边;而对于寻找最大值,总是沿着树的右边。
search(key) {
return this.searchNode(this.root, key);
};
// 用来寻找一棵树或其任意子树中的一个特定的值
searchNode(node, key) {
// 检验传入的node是否合法
if (node == null) {
return false;
}
// 要找的键比当前的节点小,则继续在左侧的子树上搜索
if (this.compareFn(key, node.key) === Compare.LESS_THAN) {
return this.searchNode(node.left, key);
}
// 要找的键比当前的节点大,从右侧子节点开始继续搜索
else if (this.compareFn(key, node.key) === Compare.BIGGER_THAN) {
return this.searchNode(node.right, key);
}
// 要找的键和当前节点相等
else {
return true;
}
}
测试方法:
console.log(tree.search(1) ? 'found' : 'not found'); // not found
console.log(tree.search(8) ? 'found' : 'not found'); // found
remove(key) {
return this.removeNode(this.root, key);
}
removeNode(node, key) {
// 如果正在检测的节点为null,说明键不存在于树中
if (node == null) {
return false;
}
// 如果要找的键比当前节点的值小,就沿树的左边找下一个节点
if (this.compareFn(key, node.key) === Compare.LESS_THAN) {
node.left = this.removeNode(node.left, key);
return node;
}
// 如果要找的键比当前节点的值大,就沿树的右边找下一个节点
else if (this.compareFn(key, node.key) === Compare.BIGGER_THAN) {
node.right = this.removeNode(node.right, key);
return node;
}
// 找到了要找的键
else {
// 第一种情况:移除一个叶节点
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;
}
// 第三种情况:移除有两个字节点的节点
// 找到要移除的节点后,需要找到它右边子树中最小的节点
const aux = this.minNode(node.right);
// 用它右侧子树中最小节点的键去更新这个节点的值
node.key = aux.key;
// 继续把右侧子树中的最小节点移除
node.right = this.removeNode(node.right, aux.key);
// 向它的父节点返回更新后节点的引用
return node;
}
}
问题:BST存在一个问题,取决于你添加的节点数,树的一条边可能会非常深;也就是说,树的一条分支会有很多层,而其他的分支却只有几层。
解决:尽可能尝试转换为完全树(Adelson-Velskii-Landi)是一种自平衡二叉搜索树,意思是任何一个节点左右两侧子树的高度之差最多为1。
创建AVLTree类:
class AVLTree extends BinarySearchTree {
constructor(compareFn = defaultCompare) {
super(compareFn);
this.compareFn = compareFn;
this.root = null;
}
}
AVL树是一个BST,可以扩展BST类,只需要覆盖用来维持AVL树平衡的方法,也就是insert、insertNode和removeNode方法。所有其他的BST方法将会被AVLTree类继承。
节点的高度是从节点到其任意子节点的边的最大值
计算节点高度:
getNodeHeight(node) {
if (node == null) {
return -1;
}
// 对每个节点计算右子树高度和左子树高度之间的差值,如果结果不是0,1或-1这三个值之一,则需要AVL树
return Math.max(
this.getNodeHeight(node.left), this.getNodeHeight(node.right)
) + 1;
}
getBalanceFactor(node) {
const heightDifference = this.getNodeHeight(node.left) - this.getNodeHeight(node.right);
switch (heightDifference) {
case -2:
return BalanceFactor.UNBALANCED_RIGHT;
case -1:
return BalanceFactor.SLIGHTLY_UNBALANCED_RIGHT;
case 1:
return BalanceFactor.SLIGHTLY_UNBALANCED_LEFT;
case 2:
return BalanceFactor.UNBALANCED_LEFT;
default:
return BalanceFactor.BALANCED;
}
}
为了避免直接在代码中处理平衡因子的数值,创建一个用来作为计数器的JavaScript常量:
const BalanceFactor = {
UNBALANCED_RIGHT: 1,
SLIGHTLY_UNBALANCED_RIGHT: 2,
BALANCED: 3,
SLIGHTLY_UNBALANCED_LEFT: 4,
UNBALANCED_LEFT: 5
}
在对AVL树添加或移除节点后,需要计算节点的高度并验证树是否需要进行平衡。
向AVL树插入节点时,可以执行单旋转或双旋转两种平衡操作。
有以下四种场景:
rotationLL(node) {
// 与平衡操作相关节点有X,Y和Z,将节点X置于节点Y(平衡因子为+2)所在的位置
// 节点X的左子树保持不变
const tmp = node.left;
// 将节点Y的左子节点置为节点X的右子节点Z
node.left = tmp.right;
// 将节点X的右子节点置为节点Y
tmp.right = node;
return tmp;
}
rotationRR(node) {
// 与平衡操作相关的节点有三个(X、Y、Z),将节点X置于节点Y(平衡因子为-2)所在的位置
// 节点X的右子树保持不变
const tmp = node.right;
// 将节点Y的右子节点置为节点X的左子节点Z
node.right = tmp.left;
// 将节点X的左子节点置为节点Y
tmp.left = node;
return tmp;
}
rotationLR(node) {
node.left = this.rotationRR(node.left);
return this.rotationLL(node);
}
rotationRL(node) {
node.right = this.rotationLL(node.right);
return this.rotationRR(node);
}
向AVL树插入节点和在BST中是一样的。除了插入节点外,我们还要验证插入后树是否还是平衡的,如果不是,就要进行必要的旋转操作。
insert(key) {
this.root = this.insertNode(this.root, key);
}
insertNode(node, key) {
// 像在BST树中一样插入节点
if (node == null) {
return new Node(key);
} else if (this.compareFn(key, node.key) === Compare.LESS_THAN) {
node.left = this.insertNode(node.left, key);
} else if (this.compareFn(key, node.key) === Compare.BIGGER_THAN) {
node.right = this.insertNode(node.right, key);
} else {
return node;
}
// 如果需要,将树进行平衡操作,先检查是否需要平衡操作
const balanceFactor = this.getBalanceFactor(node);
// 如果在向左侧子树插入节点后树不平衡
if (balanceFactor === BalanceFactor.UNBALANCED_LEFT) {
// 比较是否插入的键小于左侧子节点的键
if (this.compareFn(key, node.left.key) === Compare.LESS_THAN) {
// 如果是,进行LL旋转
node = this.rotationLL(node);
} else {
// 否则,进行LR旋转
return this.rotationLR(node);
}
}
// 如果在右侧子树插入节点后树不平衡
if (balanceFactor === BalanceFactor.UNBALANCED_RIGHT) {
// 比较是否插入的键小于右侧子节点的键
if (this.compareFn(key, node.right.key) === Compare.BIGGER_THAN) {
// 如果是,进行RR旋转
node = this.rotationRR(node);
} else {
// 否则,进行RL旋转
return this.rotationRL(node);
}
}
return node;
}
从AVL树移除节点和在BST中是一样的。除了移除节点外,我们还要验证移除后树是否还是平衡的,如果不是,就要进行必要的旋转操作。
removeNode(node, key) {
node = super.removeNode(node, key);
if (node == null) {
return node;
}
// 检查树是否需要平衡
const balanceFactor = this.getBalanceFactor(node);
if (balanceFactor === BalanceFactor.UNBALANCED_LEFT) {
const balanceFactorLeft = this.getBalanceFactor(node.left);
if (balanceFactorLeft === BalanceFactor.BALANCED || balanceFactorLeft === BalanceFactor.SLIGHTLY_UNBALANCED_LEFT) {
return this.rotationLL(node);
}
if (balanceFactorLeft === BalanceFactor.SLIGHTLY_UNBALANCED_RIGHT) {
return this.rotationLR(node.left);
}
}
if (balanceFactor === BalanceFactor.UNBALANCED_RIGHT) {
const balanceFactorRight = this.getBalanceFactor(node.right);
if (balanceFactorRight === BalanceFactor.BALANCED || balanceFactorRight === BalanceFactor.SLIGHTLY_UNBALANCED_RIGHT) {
return this.rotationRR(node);
}
if (balanceFactorRight === BalanceFactor.SLIGHTLY_UNBALANCED_LEFT) {
return this.rotationRL(node.right);
}
}
return node;
}
红黑树也是一种自平衡树。
- 进行多次插入和删除,选择红黑树比较好
- 插入和删除频率较低,选择AVL树比较好
红黑树的每个节点遵循以下规则:
红黑树也是二叉搜索树,可以扩展二叉搜索树类并重写红黑树属性需要的方法。
class RedBlackTree extends BinarySearchTree {
constructor(compareFn = defaultCompare) {
super(compareFn);
this.compareFn = compareFn;
this.root = null;
}
}
向红黑树插入了新的节点:
insert(key: T) {
// 如果树是空的
if (this.root == null) {
// 创建一个红黑树节点
this.root = new RedBlackNode(key);
// 将根节点颜色设为黑色
this.root.color = Color.BLACK;
} else {
// 像二叉搜索树一样在正确的位置插入节点
const newNode = this.insertNode(this.node, key);
// insertNode方法需要返回新插入的节点,这样可以验证在插入后,红黑树的规则是否得到了满足
this.fixTreeProperties(newNode);
}
}
对红黑树来说,节点和之前比起来需要一些额外的属性:节点的颜色和指向父节点的引用:
class RedBlackNode extends Node {
constructor(key) {
super(key);
this.key = key;
// 节点的颜色
this.color = Colors.RED;
// 指向父节点的引用
this.parent = null;
}
isRed() {
return this.color === Colors.RED;
}
}
重写insertNode方法:
insertNode(node, key) {
if (this.compareFn(key, node.key) === Color.LESS_THAN) {
if (node.left == null) {
node.left = new RedBlackNode(key);
// 保存了指向被插入节点父节点的引用
node.left.parent = node;
// 返回了节点的引用
return node.left;
} else {
return this.insertNode(node.left, key);
}
} else if (node.right == null) {
node.right = new RedBlackNode(key);
// 保存了指向被插入节点父节点的引用
node.right.parent = node;
// 返回了节点的引用
return node.right;
} else {
return this.insertNode(node.right, key);
}
}
要验证红黑树是否还是平衡的以及满足它的所有要求,需要使用两个概念:重新填色和旋转。
在向树中插入节点后,新节点将会是红色。这不会影响黑色节点数量的规则,但需要改变父节点、祖父节点和叔节点。
fixTreeProperties(node) {
while (node && node.parent && node.parent.color.isRed() && // 验证它的父节点是否是红色
node.color !== Colors.BLACK // 验证这个节点是否为黑色
) {
// 保存父节点的引用
let parent = node.parent;
// 保存祖父节点的引用
const grandParent = parent.parent;
// 情形1:父节点是左侧子节点
if (grandParent && grandParent.left === parent) {
// 要验证父节点是左侧子节点
const uncle = grandParent.right;
// 叔节点也是红色,只需要重新填色
if (uncle && uncle.color === Colors.RED) {
// 改变祖父节点、父节点和叔节点的颜色,并且将当前节点的引用指向祖父节点
grandParent.color = Colors.RED;
parent.color = Colors.BLACK;
uncle.color = Colors.BLACK;
node = grandParent;
} else {
// 节点是右侧子节点,左旋转
if (node === parent.right) {
this.rotationRR(parent);
node = parent;
parent = node.parent;
}
// 节点是左侧子节点,右旋转
this.rotationLL(grandParent);
parent.color = Colors.BLACK;
grandParent.color = Colors.RED;
node = parent;
}
}
// 父节点是右侧子节点
else {
const uncle = grandParent.left;
// 叔节点是红色,需要重新填色
if (uncle && uncle.color === Colors.RED) {
grandParent.color = Colors.RED;
parent.color = Colors.BLACK;
uncle.color = Colors.BLACK;
node = grandParent;
} else {
// 节点是左侧子节点,右旋转
if (node === parent.left) {
this.rotationLL(parent);
node = parent;
parent = node.parent;
}
// 节点是右侧子节点,左旋转
this.rotationRR(grandParent);
parent.color = Colors.BLACK;
grandParent.color = Colors.RED;
node = parent;
}
}
}
// 为了保证根节点的颜色始终是黑色,在代码最后设置根节点的颜色
this.root.color = Colors.BLACK;
}
在插入算法中,我们只使用了右-右旋转和左-左旋转。逻辑和AVL树是一样,但由于保存了父节点的引用,需要将引用更新为旋转后的新父节点。
rotationLL(node) {
const tmp = node.left;
node.left = tmp.right;
if (tmp.right && tmp.right.key) {
tmp.right.parent = node;
}
tmp.parent = node.parent;
if (!node.parent) {
this.root = tmp;
} else {
if (node === node.parent.left) {
node.parent.left = tmp;
} else {
node.parent.right = tmp;
}
}
tmp.right = node;
node.parent = tmp;
}
rotationRR(node) {
const tmp = tmp.right;
node.right = tmp.left;
if (tmp.left && tmp.left.key) {
tmp.left.parent = node;
}
tmp.parent = node.parent;
if (!node.parent) {
this.root = tmp;
} else {
if (node === node.parent.left) {
node.parent.left = tmp;
} else {
node.parent.right = tmp;
}
}
tmp.left = node;
node.parent = tmp;
}