TreeMap的源码学习
TreeMap的源码学习
一)、TreeMap的特点
根据key值进行排序。
二)、按key值排序的两种排序算法实现
1).在构造方法中传入比较器
public TreeMap(Comparator<? super K> comparator) {
this.comparator = comparator;
}
比较器comparator实现Comparator接口,实现compare(T o1, T o2)方法;
int compare(T o1, T o2);
2).key值实现Comparable接口,重写 compareTo(key k)方法
T implements Comparable<T>{
@override
public int compareTo(T k)
}
注: 对于TreeMap而言,排序是一个必须进行的过程,要正常使用TreeMap必须制 定排序规则,加入Comparator或key对象实现Comparable接口,若没有制定 比较规则则会抛出java.lang.ClassCastException。
因为String对象默认实现了Comparable接口,实现了comparaTo方法,使用String类型作为key值,不用进行排序。
三)、TreeMap的数据结构
红黑树:Entry<k, v>
Entry<k ,v> 对象的属性:
K key;
V value;
Entry<k, v> left;
Entry<k, v> right;
Entry<k, v> parent;
boolean color = BLACK;
红黑树是一种平衡查找树.
四)、TreeMap的主要属性
//比较器
private final Comparator<? super K> comparator;
//树的根节点
private transient Entry<K,V> root;
//节点个数即元素个数
private transient int size = 0;
//修改次数
private transient int modCount = 0;
//用于构造红黑树
private static final boolean RED = false;
private static final boolean BLACK = true;
五)、TreeMap的put(K key, V value)方法
步骤:
1).获取root节点,判断是否第一次添加元素 ,若是第一次添加元素则将该元素设为根节点。
2).如若不是第一次添加,根据key值,调用比较规则从根节点开始比较左右子树直至到达叶子节点,根据key的比较结果决定放置到叶子节点的左树位置或右树位置,> 0放置右树, < 0 放置右树。
3).fixAfterInsertion修改插入元素后树的结构,转为红黑树。
TreeMap<String, String> treeMap = new TreeMap<>();
treeMap.put("a","v");
public V put(K key, V value) {
//获取根节点
Entry<K,V> t = root;
//若根节点为空,直接将元素设为树的根节点,不用遍历左右子树
if (t == null) {
compare(key, key); // type (and possibly null) check
//将元素设为树的根节点
root = new Entry<>(key, value, null);
size = 1;
modCount++;
return null;
}
//根据排序规则将元素插入到树结构中
int cmp;
Entry<K,V> parent;
// split comparator and comparable paths
Comparator<? super K> cpr = comparator;
//判断是否加入比较器,如加入了比较器,使用比较器的插入规则
if (cpr != null) {
do {
parent = t;
//使用比较器的插入规则
cmp = cpr.compare(key, t.key);
if (cmp < 0)
t = t.left;
else if (cmp > 0)
t = t.right;
else
return t.setValue(value);
} while (t != null);
}
//使用key 实现Comparable的compareTo()规则
else {
if (key == null)
throw new NullPointerException();
@SuppressWarnings("unchecked")
/**
如果Key没有加入比较器或没有实现Comparable这里就
抛出java.lang,ClassCastException
*/
Comparable<? super K> k = (Comparable<? super K>) key;
//一直比较左右子树的key值,直至到达叶子节点
do {
parent = t;
cmp = k.compareTo(t.key);
if (cmp < 0)
t = t.left;
else if (cmp > 0)
t = t.right;
else
return t.setValue(value);
} while (t != null);
}
//根据返回的比较结果决定插入节点的位置
Entry<K,V> e = new Entry<>(key, value, parent);
//< 0 作为节点的左子树
if (cmp < 0)
parent.left = e;
// > 0 作为节点的右子树
else
parent.right = e;
//调整插入元素的树结构,转为红黑树
fixAfterInsertion(e);
size++;
modCount++;
return null;
}
六)、TreeMap的get(K key)方法
步骤:
1).获取根节点
2).从根节点开始比较左右子树的key值
3).若树中某一节点的key值与查找的key值相等,则返回对应的Entry<k, v>对象
treeMap.get("a");
public V get(Object key) {
Entry<K,V> p = getEntry(key);
return (p==null ? null : p.value);
}
final Entry<K,V> getEntry(Object key) {
// Offload comparator-based version for sake of performance
//判断是否存在比较器,若有比较器,则使用比较器的比较规则
if (comparator != null)
return getEntryUsingComparator(key);
if (key == null)
throw new NullPointerException();
@SuppressWarnings("unchecked")
//没有比较器使用comparale的compareTo()规则
Comparable<? super K> k = (Comparable<? super K>) key;
Entry<K,V> p = root;
//从根节点遍历左右子树,若key值相等,则返回当前节点对象
while (p != null) {
int cmp = k.compareTo(p.key);
if (cmp < 0)
p = p.left;
else if (cmp > 0)
p = p.right;
else
return p;
}
return null;
}
注:TreeMap的功能比HashMap的功能强大, 实现了sortedMap接口,可以对元素 进行排序,单TreeMap的性能却略低于HashMap,当需要对集合进行排序操作可使用TreeMap.
TreeMap的源码学习的更多相关文章
- Java集合专题总结(1):HashMap 和 HashTable 源码学习和面试总结
2017年的秋招彻底结束了,感觉Java上面的最常见的集合相关的问题就是hash--系列和一些常用并发集合和队列,堆等结合算法一起考察,不完全统计,本人经历:先后百度.唯品会.58同城.新浪微博.趣分 ...
- Dubbo源码学习--集群负载均衡算法的实现
相关文章: Dubbo源码学习文章目录 前言 Dubbo 的定位是分布式服务框架,为了避免单点压力过大,服务的提供者通常部署多台,如何从服务提供者集群中选取一个进行调用, 就依赖Dubbo的负载均衡策 ...
- HashMap的源码学习以及性能分析
HashMap的源码学习以及性能分析 一).Map接口的实现类 HashTable.HashMap.LinkedHashMap.TreeMap 二).HashMap和HashTable的区别 1).H ...
- jQuery源码学习感想
还记得去年(2015)九月份的时候,作为一个大四的学生去参加美团霸面,结果被美团技术总监教育了一番,那次问了我很多jQuery源码的知识点,以前虽然喜欢研究框架,但水平还不足够来研究jQuery源码, ...
- MVC系列——MVC源码学习:打造自己的MVC框架(四:了解神奇的视图引擎)
前言:通过之前的三篇介绍,我们基本上完成了从请求发出到路由匹配.再到控制器的激活,再到Action的执行这些个过程.今天还是趁热打铁,将我们的View也来完善下,也让整个系列相对完整,博主不希望烂尾. ...
- MVC系列——MVC源码学习:打造自己的MVC框架(三:自定义路由规则)
前言:上篇介绍了下自己的MVC框架前两个版本,经过两天的整理,版本三基本已经完成,今天还是发出来供大家参考和学习.虽然微软的Routing功能已经非常强大,完全没有必要再“重复造轮子”了,但博主还是觉 ...
- MVC系列——MVC源码学习:打造自己的MVC框架(二:附源码)
前言:上篇介绍了下 MVC5 的核心原理,整篇文章比较偏理论,所以相对比较枯燥.今天就来根据上篇的理论一步一步进行实践,通过自己写的一个简易MVC框架逐步理解,相信通过这一篇的实践,你会对MVC有一个 ...
- MVC系列——MVC源码学习:打造自己的MVC框架(一:核心原理)
前言:最近一段时间在学习MVC源码,说实话,研读源码真是一个痛苦的过程,好多晦涩的语法搞得人晕晕乎乎.这两天算是理解了一小部分,这里先记录下来,也给需要的园友一个参考,奈何博主技术有限,如有理解不妥之 ...
- 我的angularjs源码学习之旅2——依赖注入
依赖注入起源于实现控制反转的典型框架Spring框架,用来削减计算机程序的耦合问题.简单来说,在定义方法的时候,方法所依赖的对象就被隐性的注入到该方法中,在方法中可以直接使用,而不需要在执行该函数的时 ...
随机推荐
- {每日一题}:tcp协议实现简单的文件下载器(单任务版)
文件下载器客户端 这个版本的只是为了方便回顾一下TCP客服端,服务端的创建流程,缺点就是 服务器一次只能让一个人访问下载,过两个写个使用面向对象写一个多线程版的强化一下. from socket i ...
- 开源造轮子:一个简洁,高效,轻量级,酷炫的不要不要的canvas粒子运动插件库
一:开篇 哈哈哈,感谢标题党的莅临~ 虽然标题有点夸张的感觉,但实际上,插件库确实是简洁,高效,轻量级,酷炫酷炫的咯.废话不多说,先来看个标配例子吧: (codepen在线演示编辑:http://co ...
- js奥义:原型与原型链(1)
要弄懂原型链,首先应先明白prototype原型对象.__proto__.对象三者之间的关系. 引入构造函数的相关定义: 构造函数是一种比较特殊的函数,用于批量实例化对象.通俗一点说,构造函数是用于生 ...
- java script三大组成部分
JavaScript是一种属于网络的脚本语言,已经被广泛用于Web应用开发,常用来为网页添加各式各样的动态功能,为用户提供更流畅美观的浏览效果.通常JavaScript脚本是通过嵌入在HTML中来实现 ...
- 你必须知道的容器监控 (3) Prometheus
本篇已加入<.NET Core on K8S学习实践系列文章索引>,可以点击查看更多容器化技术相关系列文章.上一篇介绍了Google开发的容器监控工具cAdvisor,但是其提供的操作界面 ...
- Mui 长按保存图片
必须先要 引入 mui.js,然后参考具体代码 <!DOCTYPE html> <html> <head> <meta charset="utf-8 ...
- crontab中部署Python脚本注意事项
有时候手工执行Python脚本跑的好好的,但是部署到Linux的crontab中后,就会遇到一些问题,最近终于有空整理一下这方面的内容,其实也是自己也踩了一些别人踩过的坑!这里仅仅列举个人遇到的一些小 ...
- 《Effective Java》 读书笔记(六)避免创建不必要的对象
java 有很多修饰类的属性的关键字:常用的static,final 说说final和static吧,平时在编程的时候,这两个关键字很多时候都觉得可有可无,最多的时候就是他们俩同时出现----定义常量 ...
- MIT线性代数:8.求解Ax=b,可解性和结构
- Kong03-Nginx、OpenResty、Kong 的基本概念和区别联系
Nginx.OpenRestry.Kong 这三个项目关系比较紧密: Nginx 是模块化设计的反向代理软件,C语言开发: OpenResty 是以 Nginx 为核心的 Web 开发平台,可以解析执 ...