二叉排序树(Java)
2016-07-28 00:53
543 查看
二叉排序树的删除代码,在网上看到很多,都感觉有点繁杂难以理解,于是,我结合了TreeMap的remove()方法写出二叉排序树的实现,如果有错误的地方,还请大家多多指正~~
若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值。
若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值。
它的左、右子树也分别为二叉排序树。
最好的情况是二叉排序树的形态和折半查找的判定树相同,其平均查找长度和logn成正比(O(log2(n)))。
最坏情况下,当先后插入的关键字有序时,构成的二叉排序树为一棵斜树,树的深度为n,其平均查找长度为(n + 1) / 2。也就是时间复杂度为O(n),等同于顺序查找。
因此,如果希望对一个集合按二叉排序树查找,最好是把它构建成一棵平衡的二叉排序树(平衡二叉树)。
性质
二叉排序树又称“二叉查找树”、“二叉搜索树”。二叉排序树:或者是一棵空树,或者是具有下列性质的二叉树:若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值。
若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值。
它的左、右子树也分别为二叉排序树。
代码实现
package com.xqq.二叉排序树BST; import java.util.Stack; public class MyBST { private Node head; private int size; public MyBST() { } public MyBST(int value) { head = new Node(value); size++; } public MyBST(int[] values) { size += values.length; for (int i = 0; i < values.length; i++) { insert(values[i]); } } /** * 插入节点 */ public void insert(int value) { if (head == null) { head = new Node(value); } else { insert(head, value); } } /** * 递归插入元素 */ public void insert(Node head, int value) { if (value <= head.value) { if (head.left == null) { head.left = new Node(value); size++; } else { insert(head.left, value); } } else { if (head.right == null) { head.right = new Node(value); size++; } else { insert(head.right, value); } } } /** * 检查是否包含该元素 */ public boolean contains(int value) { Node curr = head; while(curr != null){ if(curr.value > value){ curr = curr.left; }else if(curr.value < value){ curr = curr.right; }else { return true; } } return false; } /** * 移除节点:分3种情况 * 1. 该节点为叶子节点,直接让其父节点指向的该节点置为空 * 2. 该节点有一个孩子节点,则让其父节点指向其孩子节点 * 3. 该节点有两个孩子,则让其右孩子节点的最左节点与该节点交换位置, * 转换为情况1或2,删除该节点 * * 注意:若该节点为根节点时,需要对head进行相应的操作 */ public boolean remove(int value){ // 查找节点 Node curr = search(value); // 节点不存在,直接返回false if(curr == null) return false; // 查找其父亲节点 Node father = getFather(value); // 如果该节点有两个孩子,则找到其右孩子节点的最左节点 if(curr.left != null && curr.right != null){ // 找到右孩子节点的最左节点 Node temp = getMostLeft(curr); // 找到其父亲节点 father = getFather(temp.value); // 将找到的该节点上移至需要删除节点的位置 curr.value = temp.value; // 准备删除找到的节点 curr = temp; } // 找到需要删除的节点替补的孩子节点 Node replacement = (curr.left != null ? curr.left : curr.right); if(replacement != null){ if(father == null){ head = replacement; }else if(father.left == curr){ father.left = replacement; }else { father.right = replacement; } curr.left = curr.right = null; }else if(father == null){//为根节点 head = null; }else {//无孩子,为叶子节点 if(father.left == curr) father.left = null; else father.right = null; } return true; } /** * 找到节点的右孩子的最左节点 */ private Node getMostLeft(Node curr) { Node p = curr.right; while(p.left != null){ p = p.left; } return p; } /** * 获取父节点 */ public Node getFather(int value){ Node curr = head; Node father = null; while(curr != null){ if(curr.value < value){ father = curr; curr = curr.right; }else if(curr.value > value){ father = curr; curr = curr.left; }else { return father; } } return null; } /** * 查找节点 */ public Node search(int value){ Node curr = head; while(curr != null){ if(curr.value < value){ curr = curr.right; }else if(curr.value > value){ curr = curr.left; }else { return curr; } } return null; } public int size() { return size; } /** * 非递归:中序遍历 */ public void inOrder(){ System.out.print("非递归中序遍历: "); Node curr = head; if(curr == null) return ; Stack<Node> stack = new Stack<Node>(); while(!stack.isEmpty() || curr != null){ if(curr != null){ stack.push(curr); curr = curr.left; }else { curr = stack.pop(); System.out.print(curr.value + " "); curr = curr.right; } } System.out.println(); } @Override public String toString() { inOrder(); return ""; } public static class Node { int value; Node left, right; public Node(int value) { this.value = value; } @Override public String toString() { return value + " "; } } } 测试代码: package com.xqq.二叉排序树BST; public class Test { public static void main(String[] args) { MyBST bst = new MyBST(new int[]{3, 2, 1, 4}); bst.toString(); System.out.println(bst.contains(2)); System.out.println(bst.remove(3)); System.out.println(bst.remove(6)); bst.toString(); } } 运行结果: 非递归中序遍历: 1 2 3 4 true true false 非递归中序遍历: 1 2 4
二叉排序树性能分析
每个结点的Ci为该结点的层次数。最好的情况是二叉排序树的形态和折半查找的判定树相同,其平均查找长度和logn成正比(O(log2(n)))。
最坏情况下,当先后插入的关键字有序时,构成的二叉排序树为一棵斜树,树的深度为n,其平均查找长度为(n + 1) / 2。也就是时间复杂度为O(n),等同于顺序查找。
因此,如果希望对一个集合按二叉排序树查找,最好是把它构建成一棵平衡的二叉排序树(平衡二叉树)。
相关文章推荐
- java代理模式和动态代理
- 安卓中的延时启动 与 JAVA中的计时器/定时器类
- 二、java中的基本数据类型
- Java接受键盘输入
- Java源码-简单的绘图板
- IllegalStateException异常处理
- java中为什么byte的取值范围是-128到+127
- 启动eclipse报错 Could not create the Java Virtual Machine
- JVM的栈上分配与逃逸分析(Escape Analysis)
- Java内存区域与异常
- java之位运算基础
- Java的I/O流
- Java中clone的用法
- Android基础:Eclipse关联V4包源码
- Java同步块(synchronized block)使用详解
- Java 消亡了?不!原因在这…
- eclipse新建workspace使用之前workspace的个性配置
- JAVA学习日志(8-2-多态成员特点与实例)
- 使用Swagger整合SpringMVC自动生成Restful接口文档
- Java源码-屏幕分辨率测试程序