java二叉查找树的实现代码

作者:evasean 时间:2022-08-12 02:36:43 

本文实例为大家分享了java二叉查找树的具体代码,供大家参考,具体内容如下


package 查找;

import edu.princeton.cs.algs4.Queue;
import edu.princeton.cs.algs4.StdOut;

public class BST<Key extends Comparable<Key>, Value> {
 private class Node {
   private Key key; // 键
   private Value value;// 值
   private Node left, right; // 指向子树的链接
   private int n; // 以该节点为根的子树中的节点总数

public Node(Key key, Value val, int n) {
     this.key = key;
     this.value = val;
     this.n = n;
   }
 }

private Node root;

public int size() {
   return size(root);
 }

private int size(Node x) {
   if (x == null)
     return 0;
   else
     return x.n;
 }

/**
  * 如果树是空的,则查找未命中 如果被查找的键小于根节点,则在左子树中继续查找 如果被查找的键大于根节点,则在右子树中继续查找
  * 如果被查找的键和根节点的键相等,查找命中
  *
  * @param key
  * @return
  */
 public Value get(Key key) {
   return get(root, key);
 }

private Value get(Node x, Key key) {
   if (x == null)
     return null;
   int cmp = key.compareTo(x.key);
   if (cmp < 0)
     return get(x.left, key);
   else if (cmp > 0)
     return get(x.right, key);
   else
     return x.value;
 }

/**
  * 二叉查找树的一个很重要的特性就是插入的实现难度和查找差不多。 当查找到一个不存在与树中的节点(null)时,new 新节点,并将上一路径指向该节点
  *
  * @param key
  * @param val
  */
 public void put(Key key, Value val) {
   root = put(root, key, val);
 }

private Node put(Node x, Key key, Value val) {
   if (x == null)
     return new Node(key, val, 1);
   int cmp = key.compareTo(x.key);
   if (cmp < 0)
     x.left = put(x.left, key, val);
   else if (cmp > 0)
     x.right = put(x.right, key, val);
   else
     x.value = val;
   x.n = size(x.left) + size(x.right); // 要及时更新节点的子树数量
   return x;
 }

public Key min() {
   return min(root).key;
 }

private Node min(Node x) {
   if (x.left == null)
     return x;
   return min(x.left);
 }

public Key max() {
   return max(root).key;
 }

private Node max(Node x) {
   if (x.right == null)
     return x;
   return min(x.right);
 }

/**
  * 向下取整:找出小于等于该键的最大键
  *
  * @param key
  * @return
  */
 public Key floor(Key key) {
   Node x = floor(root, key);
   if (x == null)
     return null;
   else
     return x.key;
 }

/**
  * 如果给定的键key小于二叉查找树的根节点的键,那么小于等于key的最大键一定出现在根节点的左子树中
  * 如果给定的键key大于二叉查找树的根节点,那么只有当根节点右子树中存在大于等于key的节点时,
  * 小于等于key的最大键才会出现在右子树中,否则根节点就是小于等于key的最大键
  *
  * @param x
  * @param key
  * @return
  */
 private Node floor(Node x, Key key) {
   if (x == null)
     return null;
   int cmp = key.compareTo(x.key);
   if (cmp == 0)
     return x;
   else if (cmp < 0)
     return floor(x.left, key);
   else {
     Node t = floor(x.right, key);
     if (t == null)
       return x;
     else
       return t;
   }
 }

/**
  * 向上取整:找出大于等于该键的最小键
  *
  * @param key
  * @return
  */
 public Key ceiling(Key key) {
   Node x = ceiling(root, key);
   if (x == null)
     return null;
   else
     return x.key;
 }

/**
  * 如果给定的键key大于二叉查找树的根节点的键,那么大于等于key的最小键一定出现在根节点的右子树中
  * 如果给定的键key小于二叉查找树的根节点,那么只有当根节点左子树中存在大于等于key的节点时,
  * 大于等于key的最小键才会出现在左子树中,否则根节点就是大于等于key的最小键
  *
  * @param x
  * @param key
  * @return
  */
 private Node ceiling(Node x, Key key) {
   if (x == null)
     return null;
   int cmp = key.compareTo(x.key);
   if (cmp == 0)
     return x;
   else if (cmp > 0) {
     return ceiling(x.right, key);
   } else {
     Node t = floor(x.left, key);
     if (t == null)
       return x;
     else
       return t;
   }
 }

/**
  * 选择排名为k的节点
  *
  * @param k
  * @return
  */
 public Key select(int k) {
   return select(root, k).key;
 }

private Node select(Node x, int k) {
   if (x == null)
     return null;
   int t = size(x.left);
   if (t > k)
     return select(x.left, k);
   else if (t < k)
     return select(x.right, k - t - 1);// 根节点也要排除掉
   else
     return x;
 }

/**
  * 查找给定键值的排名
  *
  * @param key
  * @return
  */
 public int rank(Key key) {
   return rank(key, root);
 }

private int rank(Key key, Node x) {
   if (x == null)
     return 0;
   int cmp = key.compareTo(x.key);
   if (cmp < 0)
     return rank(key, x.left);
   else if (cmp > 0)
     return 1 + size(x.left) + rank(key, x.right);
   else
     return size(x.left);
 }
 /**
  * 删除最小键值对
  */
 public void deleteMin(){
   root = deleteMin(root);
 }
 /**
  * 不断深入根节点的左子树直到遇见一个空链接,然后将指向该节点的链接指向该结点的右子树
  * 此时已经没有任何链接指向要被删除的结点,因此它会被垃圾收集器清理掉
  * @param x
  * @return
  */
 private Node deleteMin(Node x){
   if(x.left == null) return x.right;
   x.left = deleteMin(x.left);
   x.n = size(x.left)+size(x.right) + 1;
   return x;
 }

public void deleteMax(){
   root = deleteMax(root);
 }
 private Node deleteMax(Node x){
   if(x.right == null ) return x.left;
   x.right = deleteMax(x.right);
   x.n = size(x.left)+size(x.right) + 1;
   return x;
 }

public void delete(Key key){
   root = delete(root,key);
 }
 private Node delete(Node x, Key key){
   if(x == null) return null;
   int cmp = key.compareTo(x.key);
   if(cmp < 0) x.left = delete(x.left,key);
   else if(cmp > 0) x.right = delete(x.right,key);
   else{
     if(x.right == null) return x.left;
     if(x.left == null ) return x.right;
     /**
      * 如果被删除节点有两个子树,将被删除节点暂记为t
      * 从t的右子树中选取最小的节点x,将这个节点x的左子树设为t的左子树
      * 这个节点x的右子树设为t的右子树中删除了最小节点的子树,这样就成功替换了t的位置
      */
     Node t = x;
     x = min(t.right);
     x.left = t.left;
     x.right = deleteMin(t.right);
   }
   x.n = size(x.left) + size(x.right) +1;
   return x;
 }

public void print(){
   print(root);
 }
 private void print(Node x){
   if(x == null ) return;
   print(x.left);
   StdOut.println(x.key);
   print(x.right);
 }

public Iterable<Key> keys(){
   return keys(min(),max());
 }
 public Iterable<Key> keys(Key lo, Key hi){
   Queue<Key> queue = new Queue<Key>();
   keys(root, queue, lo, hi);
   return queue;
 }
 private void keys(Node x, Queue<Key> queue, Key lo, Key hi){
   if(x == null) return;
   int cmplo = lo.compareTo(x.key);
   int cmphi = lo.compareTo(x.key);
   if(cmplo < 0 ) keys(x.left,queue,lo,hi);
   if(cmplo <= 0 && cmphi >= 0) queue.enqueue(x.key);
   if(cmphi > 0 ) keys(x.right,queue,lo,hi);
 }
}

来源:http://www.cnblogs.com/evasean/p/7327278.html

标签:java,二叉查找树
0
投稿

猜你喜欢

  • SharedPreference 初始化源码解析

    2023-11-13 07:00:43
  • IDEA 2022 中的Lombok 使用基础教程

    2023-04-09 21:57:09
  • Java之String、StringBuffer、StringBuilder的区别分析

    2022-12-29 10:38:05
  • Java 方法签名详解及实例代码

    2022-02-04 05:56:06
  • Spring Security和Shiro的相同点与不同点整理

    2023-01-15 17:07:20
  • Android沉浸式顶部实现代码及效果

    2021-06-19 18:51:49
  • Java全面深入探究SpringBoot拦截器与文件上传

    2021-11-12 20:08:24
  • Android仿百度地图小度语音助手的贝塞尔曲线动画

    2022-03-21 00:09:26
  • C#执行Javascript代码的几种方法总结

    2023-06-12 10:06:14
  • 详解kafka中的消息分区分配算法

    2021-06-02 08:16:15
  • java旋转二维数组实例

    2021-07-04 05:39:48
  • Android购物车项目快速开发

    2021-09-11 01:28:07
  • Java中Lambda表达式的使用详解

    2022-01-02 04:52:16
  • JavaWeb开发之【Tomcat 环境配置】MyEclipse+IDEA配置教程

    2022-05-04 12:50:18
  • C#使用Clipboard类实现剪贴板功能

    2022-09-29 06:03:54
  • SpringBoot整合Shiro的代码详解

    2023-10-30 10:53:31
  • Android多渠道打包的方法步骤

    2022-10-17 12:05:07
  • Java简单实现定时器

    2023-07-16 18:10:58
  • 基于C#实现Windows服务状态启动和停止服务的方法

    2022-07-27 12:17:57
  • SpringBoot中使用Filter和Interceptor的示例代码

    2022-06-28 17:20:04
  • asp之家 软件编程 m.aspxhome.com