一.排序二叉树
-
排序二叉树是一种特殊结构的二叉树,可以非常方便地对树中所有节点进行排序和检索。
-
排序二叉树要么是一棵空二叉树,要么是具有下列性质的二叉树:
-
若它的左子树不空,则左子树上所有节点的值均小于它的根节点的值。
-
若它的右子树不空,则右子树上所有节点的值均大于它的根节点的值。
-
二.排序二叉树添加节点
以根节点当前节点开始搜索,拿被添加的节点的值和当前节点的值比较。
- 如果被添加的节点的值更小,则以当前节点的左子节点作为新的当前节点。
- 如果被添加的节点的值更大,则以当前节点的右子节点作为新的当前节点。
-
重复12两个步骤,直到新的当前节点为空,则此地方就是添加节点的地方。
三.排序二叉树删除节点
- 被删除的节点是叶子节点,只需将它从其父节点中删除即可。
- 被删除节点 p 只有左子树,将 p 的左子树 pL 添加成 p 的父节点的左或者右子树即可(见图2.1);被删除节点 p 只有右子树,将 p 的右子树 pR 添加成 p 的父节点的左或者右子树即可(见图2.2)。左还是右取决于 p 是其父节点 q 的左、右子节点。
- 若被删除节点 p 的左、右子树均非空,有两种做法:
- 以 p 节点的中序前趋(见图3.1.1)或后继(见图3.1.2)替代 p 所指节点,然后再从原排序二叉树中删去中序前趋或后继节点即可。(也就是用大于 p 的最小节点或小于 p 的最大节点代替 p 节点即可)。
- 将 pL 设为 p 的父节点 q 的左或右子节点(取决于 p 是其父节点 q 的左、右子节点),将 pR 设 为 p 节点的中序前趋节点 s 的右子节点(s 是 pL 最右下的节点,也就是 pL 子树中最大的节点)。(见图3.2)
四.排序二叉树检索节点
以根节点当前节点开始检索,拿被检索的节点的值和当前节点的值比较。
- 如果被检索的节点的值更小,则以当前节点的左子节点作为新的当前节点。
- 如果被检索的节点的值更大,则以当前节点的右子节点作为新的当前节点。
-
重复12两个步骤,直到被检索的节点的值和当前节点的值相等,如果找不到返回null。
五.红黑树
- 排序二叉树虽然可以快速检索,但在最坏的情况下:如果插入的节点集本身就是有序的,要么是由小 到大排列,要么是由大到小排列,那么最后得到的排序二叉树将变成链表:所有节点只有左节点(如果插 入节点集本身是大到小排列);或所有节点只有右节点(如果插入节点集本身是小到大排列)。在这种情 况下,排序二叉树就变成了普通链表,其检索效率就会很差。
- 为了改变排序二叉树存在的不足,Rudolf Bayer 与 1972 年发明了另一种改进后的排序二叉树:红黑 树,他将这种排序二叉树称为“对称二叉 B 树”,而红黑树这个名字则由 Leo J. Guibas 和 Robert Sedgewick 于 1978 年首次提出。
- 红黑树是一个更高效的检索二叉树,因此常常用来实现关联数组。典型地,JDK 提供的集合类 TreeMap 本身就是一个红黑树的实现。
-
红黑树在原有的排序二叉树增加了如下几个要求:
-
性质 1:每个节点要么是红色,要么是黑色。
-
性质 2:根节点永远是黑色的。
-
性质 3:所有的叶节点都是空节点(即 null),并且是黑色的。
-
性质 4:每个红色节点的两个子节点都是黑色。(从每个叶子到根的路径上不会有两个连续的红色节点。
-
性质 5:从任一节点到其子树中每个叶子节点的路径都包含相同数量的黑色节点。
-
-
根据性质 5:红黑树从根节点到每个叶子节点的路径都包含相同数量的黑色节点,因此从根节点到叶 子节点的路径中包含的黑色节点数被称为树的“黑色高度(black-height)”。
- 性质 4 则保证了从根节点到叶子节点的最长路径的长度不会超过任何其他路径的两倍。假如有一棵黑 色高度为 3 的红黑树:从根节点到叶节点的最短路径长度是 2,该路径上全是黑色节点(黑节点 - 黑节点 - 黑节点)。最长路径也只可能为 4,在每个黑色节点之间插入一个红色节点(黑节点 - 红节点 - 黑节点 - 红节点 - 黑节点),性质 4 保证绝不可能插入更多的红色节点。由此可见,红黑树中最长路 径就是一条红黑交替的路径。
-
由此我们可以得出结论:对于给定的黑色高度为 N 的红黑树,从根到叶子节点的最短路径长度为 N-1 ,最长路径长度为 2 * (N-1)。
-
排序二叉树的深度直接影响了检索的性能,正如前面指出,当插入节点本身就是由小到大排列 时,排序二叉树将变成一个链表,这种排序二叉树的检索性能最低:N 个节点的二叉树深度就是 N-1。
红黑树通过上面这种限制来保证它大致是平衡的——因为红黑树的高度不会无限增高,这样保证红黑 树在最坏情况下都是高效的,不会出现普通排序二叉树的情况。
由于红黑树只是一个特殊的排序二叉树,因此对红黑树上的只读操作与普通排序二叉树上的只读操作 完全相同,只是红黑树保持了大致平衡,因此检索性能比排序二叉树要好很多。
但在红黑树上进行插入操作和删除操作会导致树不再符合红黑树的特征,因此插入操作和删除操作都 需要进行一定的维护,以保证插入节点、删除节点后的树依然是红黑树。
五.红黑树插入节点后的修复
插入操作按如下步骤进行:
- 以排序二叉树的方法插入新节点,并将它设为红色。
- 进行颜色调换和树旋转(在插入操作中,红黑树的性质 1 和性质 3 两个永远不会发生改变,因此无需考虑红黑树的这两个特 性)。为了介绍的方便,我们把新插入的节点定义 为 N 节点,N 节点的父节点定义为 P 节点,P 节点的兄弟节点定义为 U 节点,P 节点父节点定义为 G 节点。
-
情形 1:新节点 N 是树的根节点,没有父节点
在这种情形下,直接将它设置为黑色以满足性质 2。
-
情形 2:新节点的父节点 P 是黑色
在这种情况下,新插入的节点是红色的,因此依然满足性质 4。而且因为新节点 N 有两个黑色叶子节 点;但是由于新节点 N 是红色,通过它的每个子节点的路径依然保持相同的黑色节点数,因此依然满足 性质 5。
-
情形 3:如果父节点 P 和父节点的兄弟节点 U 都是红色
在这种情况下,程序应该将 P 节点、U 节点都设置为黑色,并将 P 节点的父节点设为红色(用来保 持性质 5)。现在新节点 N 有了一个黑色的父节点 P。由于从 P 节点、U 节点到根节点的任何路径都必 须通过 G 节点,在这些路径上的黑节点数目没有改变(原来有叶子和 G 节点两个黑色节点,现在有叶子 和 P 两个黑色节点)。
-
情形 4:父节点 P 是红色、而其兄弟节点 U 是黑色或缺少;且新节点 N 是父节点 P 的右子节点, 而父节点 P 又是其父节点 G 的左子节点。
在这种情形下,我们进行一次左旋转对新节点和其父节点进行,接着按情形 5 处理以前的父节点 P( 也就是把 P 当成新插入的节点即可)。这导致某些路径通过它们以前不通过的新节点 N 或父节点 P 的 其中之一,但是这两个节点都是红色的,因此不会影响性质 5。
-
情形 5:父节点 P 是红色、而其兄弟节点 U 是黑色或缺少;且新节点 N 是其父节点的左子节点,而 父节点 P 又是其父节点 G 的左子节点。
在这种情形下,需要对节点 G 的一次右旋转,在旋转产生的树中,以前的父节点 P 现在是新节点 N 和节点 G 的父节点。由于以前的节点 G 是黑色,否则父节点 P 就不可能是红色,我们切换以前的父节 点 P 和节点 G 的颜色,使之满足性质 4,性质 5 也仍然保持满足,因为通过这三个节点中任何一个的 所有路径以前都通过节点 G,现在它们都通过以前的父节点 P。在各自的情形下,这都是三个节点中唯一 的黑色节点。
-
六.红黑树删除节点后的修复
与添加节点之后的修复类似的是,TreeMap 删除节点之后也需要进行类似的修复操作,通过这种修复 来保证该排序二叉树依然满足红黑树特征。大家可以参考插入节点之后的修复来分析删除之后的修复。
相关推荐
总的来说,TreeMap源码的学习可以帮助我们理解红黑树的数据结构和其实现原理,这对于优化和调试涉及大量数据的操作至关重要。同时,掌握TreeMap的内部机制也能提升我们在实际开发中使用Java集合框架的能力。
- **TreeMap/TreeSet**:基于红黑树实现,可以自动按自然顺序或指定的比较器排序元素。 - **Collections.sort()**:用于对集合进行排序,可以传入`Comparator`来指定排序规则。 #### 总结 Java集合容器是Java编程中...
红黑树(Red-Black Tree)是一种自平衡的二叉查找树,由计算机科学家Rudolf Bayer在1972年提出。...通过分析和实现Java中的红黑树,我们可以深入理解这种数据结构的工作原理及其在实际应用中的价值。
- **TreeMap**:基于红黑树的实现,支持自然排序或自定义排序。 - **Hashtable**:古老的哈希表实现,线程安全但效率低。 - **ConcurrentHashMap**:线程安全的哈希表实现,适用于多线程环境。 #### 4. Set接口的...
- 在很多数据结构中作为实现的基础,如Java中的TreeMap和TreeSet。 #### 九、堆 **堆**: - **定义**:具有特定结构的完全二叉树,可以是最大堆或最小堆。 - **特点**: - 父节点的值总是大于或小于其子节点的值...
- **TreeSet**和**TreeMap**:基于红黑树实现,保证排序性。 - **HashMap**和**LinkedHashMap**:存储键值对,HashMap无序,LinkedHashMap保持插入顺序或访问顺序。 - **ArrayList和LinkedList之间的选择**:取决...
本文将详细分析四种常用的`Map`实现类:`HashMap`, `LinkedHashMap`, `TreeMap`以及`HashTable`之间的区别。 #### 1. HashMap `HashMap`是一种基于哈希表实现的`Map`接口,提供了一个非同步的、允许使用`null`键和...
- Java标准库中的`java.util.TreeMap`和`java.util.TreeSet`类底层就是用红黑树实现的。 - 实现红黑树需要定义节点类,包含节点值、颜色、左子节点、右子节点以及父节点等属性,并实现插入、删除、旋转等方法。 6...
- **TreeMap/TreeSet**:基于红黑树的有序映射和集合,提供排序功能和O(log n)的时间复杂度。 2. 日期时间: - **Calendar**:抽象类,是日期和时间计算的基础,提供了丰富的日期时间操作。 - **Date**:表示...
在Java编程中,Map接口是用于存储键值对的数据结构,而Java提供了多种Map的实现,包括TreeMap、HashMap和ConcurrentSkipListMap。本文主要比较了这三种Map的性能,尤其是在插入和查找操作上的效率。 1. **TreeMap**...
- **特点**:`TreeSet` 实现了基于红黑树的排序集合,可以自然排序或通过自定义比较器排序。 - **用途**:适合需要保持元素有序的场景。 - **排序**:默认情况下,元素将按照其自然顺序(即实现了 `Comparable` 接口...
- **应用场景**:红黑树被广泛应用于实现关联容器,如Java中的TreeMap和TreeSet。 #### 十、2-3-4树与外部存储 - **2-3-4树**:一种特殊的多路搜索树,每个节点最多有三个子节点。 - **外部存储**:指数据存储在...
- **TreeMap**:基于红黑树的映射。 - **LinkedHashMap**:保持插入顺序的哈希映射。 - **Hashtable**:线程安全的映射。 - **IdentityHashMap**:基于对象身份而非哈希码的映射。 - **WeakHashMap**:允许键被...
TreeMap (可排序) - **实现**:基于红黑树。 - **特点**:键按自然顺序排序。 ##### 21. LinkedHashmap (记录插入顺序) - **实现**:结合了`HashMap`和`LinkedHashMap`的特点。 - **特点**:记录元素的插入顺序。 ...
- **TreeMap**:基于红黑树的有序Map,支持快速排序和查找。 - **Set**: - **HashSet**:基于HashMap实现,无序且不重复。 - **LinkedHashSet**:在HashSet基础上增加顺序,插入或访问顺序。 理解这些基础...
- **遍历**:红黑树支持前序遍历、中序遍历和后序遍历,这些遍历方式在任何二叉树中都有定义。 4. **代码实现**: - 在CLION IDE中实现红黑树,你需要创建`main.c`文件并定义红黑树的节点结构、颜色枚举、插入、...
- **树**:包括二叉树、平衡二叉树(AVL、红黑树)、B树、B+树等,Java的TreeMap和TreeSet基于红黑树实现。 - **图**:由顶点和边构成,用于表示实体间的关系,Java中没有内置的图数据结构,但可以通过集合类构建...
- **TreeMap**:基于红黑树实现,按键排序。 ### 3. Java中接口与抽象类的异同 - **相同点**: - 都不能实例化。 - 都可以包含抽象方法。 - 都可以实现代码复用。 - **不同点**: - **接口**: - 定义行为...
- `Map`接口的主要实现包括`HashMap`和`TreeMap`。 #### HashMap与HashTable的区别 - **线程安全性**:`HashTable`是线程安全的,而`HashMap`不是。 - **同步性**:`HashTable`中方法是同步的,而`HashMap`不是。...