Java集合中的TreeMap解讀
TreeMap解讀
TreeSet的底層是TreeMap。
public boolean add(E e) {
return m.put(e, PRESENT)==null;
//其中的e為我們放進(jìn)去的元素,作為key來存放的。
//而后面的value為PRESENT為一個(gè)固定的值,
}而其中的PRESENT是在treeSet里面創(chuàng)建的靜態(tài)的final類型的Object.
private static final Object PRESENT = new Object();
如果我們使用TreeMap,里面放置的是(e為Key,PRESENT為對(duì)應(yīng)的那一個(gè)值)。
我們通過具體的代碼設(shè)計(jì)如下所示:
package com.rgf.map;
import java.util.TreeMap;
@SuppressWarnings({"all"})
public class TreeMap_ {
public static void main(String[] args) {
//使用默認(rèn)的構(gòu)造器,創(chuàng)建TreeMap,是無序的,是指輸入輸出的順序不一致
TreeMap treeMap = new TreeMap();
treeMap.put("jack","杰克");
treeMap.put("tom","湯姆");
treeMap.put("kristina","克瑞斯提諾");
treeMap.put("smith","斯密斯");
System.out.println("treeMap="+treeMap);
}
}
}運(yùn)行界面如下所示:

我們查看如下所示:

我們發(fā)現(xiàn)TreeMap類里面除了提供無參的類以外,也有TreeMap(Comparator <K>)這樣子的構(gòu)造器。TreeMap可以傳入一個(gè)實(shí)現(xiàn)了 Comparator接口的一個(gè)匿名內(nèi)部類,匿名內(nèi)部類里面我們?nèi)匀豢梢匀ブ付ㄌ砑游覀兊逆I值對(duì)的這種排序規(guī)則。
我們?cè)O(shè)計(jì)排序規(guī)則進(jìn)行如下代碼所示:
package com.rgf.map;
import java.util.Comparator;
import java.util.TreeMap;
@SuppressWarnings({"all"})
public class TreeMap_ {
public static void main(String[] args) {
//使用默認(rèn)的構(gòu)造器,創(chuàng)建TreeMap,是無序的,是指輸入輸出的順序不一致
//我們按照傳入的k(String)的大小進(jìn)行排序。
TreeMap treeMap = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
//我們按照傳入的k(String)的大小進(jìn)行排序。
return ((String) o1).compareTo((String)o2);
}
});
treeMap.put("jack","杰克");
treeMap.put("tom","湯姆");
treeMap.put("Kristina","克瑞斯提諾");
treeMap.put("smith","斯密斯");
System.out.println("treeMap="+treeMap);
}
}我們運(yùn)行之后如下所示:

以上排序?yàn)閺男〉酱螅覀兛梢詫⑴判蛐薷臑閺拇蟮叫?,我們進(jìn)行修改代碼如下所示:
package com.rgf.map;
import java.util.Comparator;
import java.util.TreeMap;
@SuppressWarnings({"all"})
public class TreeMap_ {
public static void main(String[] args) {
//使用默認(rèn)的構(gòu)造器,創(chuàng)建TreeMap,是無序的,是指輸入輸出的順序不一致
//我們按照傳入的k(String)的大小進(jìn)行排序。
TreeMap treeMap = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
//我們按照傳入的k(String)的大小進(jìn)行排序。(從小到大)
return ((String) o2).compareTo((String)o1);
}
}) ;
treeMap.put("jack","杰克");
treeMap.put("tom","湯姆");
treeMap.put("kristina","克瑞斯提諾");
treeMap.put("smith","斯密斯");
System.out.println("treeMap="+treeMap);
}
}運(yùn)行界面如下所示:

我們可以按照長(zhǎng)度進(jìn)行排序(從小到大),我們將代碼修改如下所示:
package com.rgf.map;
import java.util.Comparator;
import java.util.TreeMap;
@SuppressWarnings({"all"})
public class TreeMap_ {
public static void main(String[] args) {
//使用默認(rèn)的構(gòu)造器,創(chuàng)建TreeMap,是無序的,是指輸入輸出的順序不一致
//我們按照傳入的k(String)的大小進(jìn)行排序。
TreeMap treeMap = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
//我們按照傳入的k(String)的長(zhǎng)度大小進(jìn)行排序。(從小到大)
return ((String) o1).length()-((String) o2).length();
}
}) ;
treeMap.put("jack","杰克");
treeMap.put("tom","湯姆");
treeMap.put("kristina","克瑞斯提諾");
treeMap.put("smith","斯密斯");
System.out.println("treeMap="+treeMap);
}
}我們的運(yùn)行界面如下所示:

我們也可以將排序找到長(zhǎng)度從大到小,我們將代碼修改如下所示:
package com.rgf.map;
import java.util.Comparator;
import java.util.TreeMap;
@SuppressWarnings({"all"})
public class TreeMap_ {
public static void main(String[] args) {
//使用默認(rèn)的構(gòu)造器,創(chuàng)建TreeMap,是無序的,是指輸入輸出的順序不一致
//我們按照傳入的k(String)的大小進(jìn)行排序。
TreeMap treeMap = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
//我們按照傳入的k(String)的長(zhǎng)度大小進(jìn)行排序。(從小到大)
return ((String) o2).length()-((String) o1).length();
}
}) ;
treeMap.put("jack","杰克");
treeMap.put("tom","湯姆");
treeMap.put("kristina","克瑞斯提諾");
treeMap.put("smith","斯密斯");
System.out.println("treeMap="+treeMap);
}
}運(yùn)行界面如下所示:

我們來進(jìn)行Debug,我們從如下代碼開始進(jìn)行debug:
TreeMap treeMap = new TreeMap(new Comparator()
源碼
1.構(gòu)造器
把傳入的實(shí)現(xiàn)了Comparator接口的匿名內(nèi)部類(對(duì)象),傳給了TreeMap的comparator屬性。
//他的底層仍然是TreeMap,把傳入的匿名內(nèi)部類comparator賦給TreeMap的一個(gè)屬性this.comparator.
public TreeMap(Comparator<? super K> comparator) {
this.comparator = comparator;
}我們退出去之后,進(jìn)去put方法:
2.調(diào)用put方法
第一次添加,把k-v封裝到Entry對(duì)象,放入root.

源碼如下所示;
public V put(K key, V value) {
Entry<K,V> t = root;//此時(shí)root為空,將root賦值給t.
//我們第一次進(jìn)去,所以還沒有進(jìn)行初始化,第一次初始化為空。
if (t == null) {//此時(shí)t為空,從而進(jìn)入里面代碼。
compare(key, key); // type (and possibly null) check
//比較key值是否相同,構(gòu)造器判斷規(guī)則不同,key值不同,當(dāng)前規(guī)則為比較長(zhǎng)度大小。則長(zhǎng)度為key.
//TreeMap底層為Entry,Entry為TreeMap里面的一個(gè)內(nèi)部類。
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);
}我們進(jìn)行第一次添加之后,我們?nèi)缦滤荆?/p>

以后添加,即要使用比較器進(jìn)行比較后添加。
此時(shí)root里面已經(jīng)有值了。此時(shí)添加第一個(gè)值得時(shí)候,沒有添加比較器。我們繼續(xù)進(jìn)行添加,查看源碼。
public V put(K key, V value) {
Entry<K,V> t = root;
//由于此前已經(jīng)有一個(gè)值了,此時(shí)root不為null.t也不為空,即不再進(jìn)入下面第一個(gè)if語句。
if (t == null) {
//第一次進(jìn)去也調(diào)用了compare方法,即動(dòng)態(tài)綁定了我們的匿名內(nèi)部類的compare方法,傳入的是兩個(gè)key,即第一次添加的key加了兩份進(jìn)去。目的是為了判定是否為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
//由于已經(jīng)有一個(gè)值了,所以我們會(huì)將比較器拿到,進(jìn)入以下代碼。
Comparator<? super K> cpr = comparator;//比較器傳入的值賦值給cpr
if (cpr != null) {
do { //遍歷所有的key,給當(dāng)前key找到適當(dāng)?shù)奈恢?
parent = t; //我們傳入的comparator賦值給parent
cmp = cpr.compare(key, t.key); //動(dòng)態(tài)綁定到我們的匿名內(nèi)部類的compare方法。
if (cmp < 0)
t = t.left;
else if (cmp > 0)
t = t.right;
else //如果遍歷過程中,發(fā)現(xiàn)準(zhǔn)備添加key和當(dāng)前已有的key相等,就不添加了,但是會(huì)進(jìn)行覆蓋原來的值。是由我們自定義的compare方法來決定的。
return t.setValue(value);
} while (t != null);
}我們會(huì)動(dòng)態(tài)綁定到我們的匿名內(nèi)部類:

我們調(diào)用了compare方法了:

我們繼續(xù)往進(jìn)走:

final int compare(Object k1, Object k2) {
//如果comparator不等于null的時(shí)候,我們進(jìn)行動(dòng)態(tài)綁定到我們的匿名內(nèi)部類的compare方法。
return comparator==null ? ((Comparable<? super K>)k1).compareTo((K)k2)
: comparator.compare((K)k1, (K)k2);
}我們會(huì)動(dòng)態(tài)綁定到我們的匿名內(nèi)部類:

此時(shí)的o1和o2都是jack.但是這個(gè)結(jié)果對(duì)是否添加當(dāng)前的value值有任何影響。主要是為了檢測(cè)是否為null。如果為null,則會(huì)在comparator方法里面拋出異常。

這里的compare作用是查看key是否為空,還有就是判斷當(dāng)前key是否實(shí)現(xiàn)了compareable接口是否是可以進(jìn)行比較的。
我們進(jìn)行排序的示例如下所示:
package com.rgf.jihe;
import org.junit.Test;
import java.util.*;
/**
* 向TreeMap中添加key-value,要求key必須是由同一個(gè)類創(chuàng)建的對(duì)象
* 因?yàn)橐凑誯ey進(jìn)行排序:自然排序、定制排序
*
*/
public class TreeMapTest {
public static void main(String[] args) {
//自然排序
TreeMap map = new TreeMap();
User u1 = new User("Tom",23);
User u2 = new User("Jerry",32);
User u3 = new User("Jack",20);
User u4 = new User("Rose",18);
map.put(u1,98);
map.put(u2,89);
map.put(u3,76);
map.put(u4,100);
Set set = map.entrySet();
Iterator iterator = set.iterator();
while (iterator.hasNext()){
Object next = iterator.next();
System.out.println(next);
}
}
//定制排序
@Test
public void test2(){
TreeMap map = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof User && o2 instanceof User){
User user=(User) o1;
User user1=(User) o2;
return Integer.compare(user.getAge(),user1.getAge());
}
throw new RuntimeException("輸入的類型不匹配");
}
});
User user = new User("rgf",123);
User user1 = new User("woer",456);
User user2 = new User("fed",789);
map.put(user1,55);
map.put(user,66);
map.put(user2,77);
Set set = map.entrySet();
Iterator iterator = set.iterator();
while (iterator.hasNext()){
Object next = iterator.next();
System.out.println(next);
}
}
}
class User implements Comparable{
private String name;
private int age;
public User() {
}
public User(String name, int age) {
this.name = name;
this.age = age;
}
//我們寫出來該方法則不會(huì)再繼續(xù)報(bào)錯(cuò)。
//按照姓名從大到小排列,年齡從小到大排列
@Override
public int compareTo(Object o) {
if(o instanceof User){
User User=(com.rgf.jihe.User)o;
//return -this.name.compareTo(user.name);為了可以確保相同的值可以進(jìn)行排序,我們進(jìn)行二級(jí)排序
int compare=-this.name.compareTo(User.name);
if(compare!=0){//如果相等,則我們進(jìn)行比較年齡
return compare;
}else {
return Integer.compare(this.age,User.age);
}
}else {
throw new RuntimeException("輸入的類型不匹配");
}
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
User User = (com.rgf.jihe.User) o;
if (age != User.age) return false;
return Objects.equals(name, User.name);
}
@Override
public int hashCode() {
int result = name != null ? name.hashCode() : 0;
result = 31 * result + age;
return result;
}
}
運(yùn)行之后如下所示:

到此這篇關(guān)于Java集合中的TreeMap解讀的文章就介紹到這了,更多相關(guān)TreeMap解讀內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
RestFul風(fēng)格 — 使用@PathVariable傳遞參數(shù)報(bào)錯(cuò)404的解決
這篇文章主要介紹了RestFul風(fēng)格 — 使用@PathVariable傳遞參數(shù)報(bào)錯(cuò)404的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
使用Spring Boot輕松實(shí)現(xiàn)流式AI輸出的步驟
本文介紹了如何使用Spring Boot和WebFlux實(shí)現(xiàn)流式AI輸出,通過非阻塞I/O、反應(yīng)式編程和函數(shù)式路由等技術(shù),優(yōu)化了AI應(yīng)用的響應(yīng)速度,提升了用戶體驗(yàn),感興趣的朋友一起看看吧2025-02-02
Spring中SmartLifecycle和Lifecycle的作用和區(qū)別
這篇文章主要介紹了Spring中SmartLifecycle和Lifecycle的作用和區(qū)別,本文通過實(shí)例代碼給大家介紹的非常詳細(xì)對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03
Java設(shè)計(jì)模式之模版方法模式簡(jiǎn)介
這篇文章主要介紹了Java設(shè)計(jì)模式之模版方法模式,需要的朋友可以參考下2014-07-07
幾句話說清session,cookie和token的區(qū)別及說明
這篇文章主要介紹了幾句話說清session,cookie和token的區(qū)別及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
IDEA如何自動(dòng)生成serialVersionUID的設(shè)置
這篇文章主要介紹了IDEA如何自動(dòng)生成 serialVersionUID 的設(shè)置,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09

