二叉树终极教程--BinarySearchTree
BinarySearchTreeMap的实现
public interface Map<K extends Comparable<K>, V> {
void put(K k, V v);
V get(K k);
void delete(K k);
boolean contains(K k);
boolean isEmpty();
int size();
int size(K lo, K hi);
K min();
K max();
K floor(K k);
K ceiling(K k);
// the number of keys less than key
int rank(K k);
K select(int k);
void deleteMin();
void deleteMax();
// keys in [lo , hi] in sorted order
Iterable<K> keys(K lo, K hi);
Iterable<K> keys();
}
Map Interface
二叉树的定义
在计算机科学中,二叉树是每个节点最多有两个子树的树结构。通常子树被称作“左子树”(left subtree)和“右子树”(right subtree)。二叉树常被用于实现二叉查找树和二叉堆。二叉树的左子节点 < 父节点 < 右子节点
这是typical的二叉树的样子, null 代表子节点为空,从这张图可以看出,左子节点 9 小于 父节点 10 小于 右子节点
private class Node<K, V> {
private K k;
private V v;
private Node<K, V> left;
private Node<K, V> right;
private int size;
Node(K k, V v) { this.k = k; this.v = v; }
Node(K k, V v, int size) { this.k = k; this.v = v; this.size = size;}
}
Node(节点)对象
二叉树的插入操作
假设我们依次插入 10 , 9, 15, 5 , 7 这5个元素到二叉树中。see what will happen 这是个动态图

@Override
public void put(K k, V v) {
root = put(root, k, v); //root 是根节点
} private Node<K, V> put(Node<K, V> node, K k, V v) {
if (node == null) return new Node<>(k, v, 1);
int cmp = node.k.compareTo(k);
if (cmp > 0) { //node的k大一点 放到左边的数中
node.left = put(node.left, k, v);
} else if (cmp < 0) { //node的k小一点 放到右边的数中
node.right = put(node.right, k, v);
} else node.v = v; node.size = size(node.left) + size(node.right) + 1;
return node;
}
put operation (插入)
二叉树的get 方法
get方法简单来说就是要找到那个key相同的对象。比如我们要在「10 , 9, 15, 5 , 7 」上图所示中找到 7

@Override
public V get(K k) {
return get(root, k);
} private V get(Node<K, V> node, K k) { if (node == null) return null; //not find
else if (node.k.compareTo(k) > 0) { //node的k大一点 放到左边的数中
return get(node.left, k);
} else if (node.k.compareTo(k) < 0) { //node的k小一点 放到右边的数中
return get(node.right, k);
} else { //equal
return node.v;
} }
get operation
二叉树的删除操作
其实想象一下,当你删除一个node的时候,你需要找一个替代node来代替这个node。
这里又分3种情况。首先假设你有如下的树结构

1.第一种情况是这个删除的节点的左右节点都是null。
比如我要删除3节点。其实只要直接把3节点reset 为null 就可以了。变成如下

2.第二种情况是删除的节点的2个子节点中有一个子节点为null
比如我要删除15。 15 的左节点是12 右节点是 null,所以符合这个情况
这个时候只需要直接把需要删除的节点 reset 为 非空的子节点就可以了
所以在这里只需要把15的值替代为12

3.第三种情况是删除的节点的2个子节点都不为null,
这个时候其实可以有2个选择,一个是把删除的节点替换为右子节点为根节点的那个树中最小的节点
比如我要删除10, 右节点为15(二叉树的删除操作的那个图,不是上面的那个图),15这个节点为根节点的树中总共有2个元素(15和12),12是最小的。所以把需要删除的节点替换为12。删除后如下

另外一种选择是把左节点为根节点的树中最大的值取出来,把需要删除的那个节点替换为这个左节点最大的元素(2个选择没什么区别)
@Override
public void delete(K k) {
delete(root, k);
}
//delete the k in the node tree and reset the size prorperty of this tree and subtrees to correct value
private Node<K, V> delete(Node<K, V> node, K k) {
if (node == null) return null; //没有找到这个node int cmp = node.k.compareTo(k);
if (cmp > 0) {
node.left = delete(node.left, k);
node.size = size(node.left) + size(node.right) + 1;
return node;
} else if (cmp < 0) {
node.right = delete(node.right, k);
node.size = size(node.left) + size(node.right) + 1;
return node;
} else { //hit the key
if (node.right == null) //if the right node is null then just replace this node with left node
return node.left;
else if (node.left == null) // if the left node is null then just replace this node with right node
return node.right;
else {
return deleteMin(node.right); // if both the subnodes are not null replace this node with the smallest node in the right sub node
}
}
} //删除从参数node开始的最小的node
private Node<K, V> deleteMin(Node<K, V> node) {
return delete(node, min(node));
} private Node<K, V> deleteMax(Node<K, V> node) {
return delete(node, max(node));
} @Override
public void deleteMin() {
deleteMin(root);
} @Override
public void deleteMax() {
deleteMax(root);
} @Override
public K min() {
return min(root);
} //get the smallest node in the given node
private K min(Node<K, V> node) {
if (node == null) return null;
for (; node.left != null; node = node.left);
return node.k;
} @Override
public K max() {
return max(root);
}
//get the most max node in the given node
private K max(Node<K, V> node) {
if (node == null) return null;
for (node = root; node.right != null; node = node.right);
return node.k;
}
delete operation 删除操作
分析
BinarySearchTree 有一个最大的缺点,就是如果插入的元素是ordered,比如我插入 1 2 3 4 5 6 这样子,元素都会排在一边。这样子查找起来路径很长,效率很低。
如果插入的元素是随机的,那么所有的get put 操作的时间复杂度应该是 和 log2(N) 成正比的
具体的实现可以参考这个。https://github.com/Cheemion/algorithms/blob/master/src/com/algorithms/tree/BinarySearchTreeMap.java
有什么错误的地方欢迎大家指正哈
二叉树终极教程--BinarySearchTree的更多相关文章
- Java 日志框架终极教程
概述 对于现代的 Java 应用程序来说,只要被部署到真实的生产环境,其日志的重要性就是不言而喻的,很难想象没有任何日志记录功能的应用程序被运行于生产环境中.日志 API 所能提供的功能是多种多样的, ...
- windows下面安装Python和pip终极教程
在大二的时候接触过一段时间的Python,最近又开始玩起了这门语言.总的来说,个 人很喜欢Python的语言风格,但是这门语言对于windows并不算很友好,因为如果是初学者在windows环境下安装 ...
- 转: windows下面安装Python和pip终极教程
原文: http://www.cnblogs.com/yuanzm/p/4089856.html 因为如果是初学者在windows环境下安装,简直是折磨人,会遇到各种蛋疼的情况.本文希望提供傻瓜式的教 ...
- MongoDB数据库安装及配置环境终极教程(windows10系统)
本文是笔者花时间踩坑踩生气了写出来的!转载请注明出处@http://www.cnblogs.com/tim100/!请尊重我的劳动成果!谢谢! 今天,给大家说说在windows10系统下MongoDB ...
- APK Multi-Tool强大的APK反编译工具终极教程
一.APK Multi-Tool介绍 APK Multi-Tool 是APK Manager的升级版,是一个强大的APK反编译工具,集多种功能于一身,是居家必备.做ROM必选的工具! 这是 ...
- 终极教程【zhong】
just for a better future! 资源教程 aiim 综合类 前端知识体系 前端知识结构 Web前端开发大系概览 We ...
- 2020年B2B外贸建站的终极教程
本文目标:按照本建站教程的顺序操作,能够实现:基于全球份额最大的建站系统“wordpress”,从零搭建一个B2B外贸网站,且建站成本每年小于1000元(如果不计算自己投入的人力成本的话). 模板站点 ...
- Docker + Jenkins + Gitlab + Pytest + Allure 接口自动化测试之持续集成实战终极教程
实战教程篇 前言 这边就不教大家怎么用 pytest 写项目了哦,下面有系列文章能帮助你快速入门 Pytest + Allure 这一篇教程主要是教如何从 0 到 1 搭建自动化测试的持续集成环境 后 ...
- Dreamweaver架设网站终极教程
转自:http://www.cnblogs.com/loveme513/archive/2006/04/03/365969.html 一.定义站点 1. 在任意一个根目录下创建好一个文件夹(我们这里假 ...
随机推荐
- JS对JSON的操作总结
对于前端完全是菜鸟,迫于无奈,工作中要用到JS,尤其对JSON的处理为多,网上搜了一下,所讲的基本雷同.所以把平时用的比较多的JSON处理方法总结了一下,权当加深记忆. 一.概述 JSON(JavaS ...
- html页面不使用缓存的代码
有时候,在我们用动态php页面调用html的时候会出现html缓存现象,所以可以在html头部加入以下代码,html就不会出现缓存了. <meta HTTP-EQUIV="pragma ...
- c# 后台get post请求
//get请求 public static TResult Get<TResult>(string host, string url) { var httpClient = new Htt ...
- Linux常见命令(五)——rmdir
前 言 JRedu 今天我们来介绍第五个命令:rmdir . 命令英文原意:remove empty directories 命令用途: rmdir:删除空目录,非空的目录不能删除 本章内容将详 ...
- 永久关闭selinux | 防火墙
关闭SELinux的两种方法 1 永久方法 – 需要重启服务器 修改/etc/selinux/config文件中设置SELINUX=disabled ,然后重启服务器. 2 临时方法 – 设置系统参数 ...
- python+selenium自动化软件测试(第2章):WebDriver API
2.1 操作元素基本方法 前言前面已经把环境搭建好了,从这篇开始,正式学习selenium的webdriver框架.我们平常说的 selenium自动化,其实它并不是类似于QTP之类的有GUI界面的可 ...
- 即时通信系统Openfire分析之四:消息路由
两个人的孤独 两个人的孤独,大抵是,你每发出去一句话,都要经由无数网络.由几百个计算机处理后,出在他的面前,而他就在你不远处. 连接管理之后 Openfire使用MINA网络框架,并设置Connect ...
- 深入理解计算机系统(1.1)------Hello World 是如何运行的
上一篇序章我谈了谈 程序员为啥要懂底层计算机结构 ,有人赞同也有人反对,但是这并不影响 LZ 对深入理解计算机系统研究的热情.这篇博客以案例驱动的模式,通过跟踪一个简单 Hello World 程序的 ...
- 求N个元素的子集合个数
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt406 一个集合有n个元素,请问怎么算出来它的子集(包括空集和本身)是 2的n ...
- [转载]历上最强的音乐播放器(jetAudio-8.0.5.320-Plus-VX
原文地址:历上最强的音乐播放器(jetAudio-8.0.5.320-Plus-VX-完全汉化版)下载作者:盖世天星 历上最强的音乐播放器(jetAudio-8.0.5.320-Plus-VX-完全汉 ...